exercise.ajax.php 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508
  1. <?php
  2. /* For licensing terms, see /license.txt */
  3. /**
  4. * Responses to AJAX calls
  5. */
  6. require_once '../../exercice/exercise.class.php';
  7. require_once '../../exercice/question.class.php';
  8. require_once '../../exercice/answer.class.php';
  9. require_once '../global.inc.php';
  10. require_once '../../exercice/exercise.lib.php';
  11. api_protect_course_script(true);
  12. $action = $_REQUEST['a'];
  13. $course_id = api_get_course_int_id();
  14. if ($debug) {
  15. error_log("$action ajax call");
  16. }
  17. $session_id = isset($_REQUEST['session_id']) ? intval($_REQUEST['session_id']) : api_get_session_id();
  18. $course_code = isset($_REQUEST['cidReq']) ? $_REQUEST['cidReq'] : api_get_course_id();
  19. if (!empty($course_code)) {
  20. $courseInfo = api_get_course_info($course_code);
  21. if ($courseInfo) {
  22. $_SESSION['cidReq'] = $course_code;
  23. $_SESSION['_real_cid'] = $courseInfo ['real_id'];
  24. $course_id = $courseInfo['real_id'];
  25. }
  26. }
  27. if (!empty($session_id)) {
  28. $_SESSION['id_session'] = $session_id;
  29. }
  30. switch ($action) {
  31. case 'get_live_stats':
  32. if (!api_is_allowed_to_edit(null, true)) {
  33. break;
  34. }
  35. // 1. Setting variables needed by jqgrid
  36. $action = $_GET['a'];
  37. $exercise_id = intval($_GET['exercise_id']);
  38. $page = intval($_REQUEST['page']); //page
  39. $limit = intval($_REQUEST['rows']); //quantity of rows
  40. $sidx = $_REQUEST['sidx']; //index to filter
  41. $sord = $_REQUEST['sord']; //asc or desc
  42. if (!in_array($sord, array('asc','desc'))) {
  43. $sord = 'desc';
  44. }
  45. // get index row - i.e. user click to sort $sord = $_GET['sord'];
  46. // get the direction
  47. if (!$sidx) {
  48. $sidx = 1;
  49. }
  50. $track_exercise = Database::get_statistic_table(TABLE_STATISTIC_TRACK_E_EXERCICES);
  51. $user_table = Database::get_main_table(TABLE_MAIN_USER);
  52. $track_attempt = Database::get_statistic_table(TABLE_STATISTIC_TRACK_E_ATTEMPT);
  53. $minutes = intval($_REQUEST['minutes']);
  54. $now = time() - 60 * $minutes;
  55. $now = api_get_utc_datetime($now);
  56. $where_condition = " orig_lp_id = 0 AND exe_exo_id = $exercise_id AND start_date > '$now' ";
  57. $sql = "SELECT COUNT(DISTINCT exe_id) FROM $track_exercise WHERE $where_condition ";
  58. $result = Database::query($sql);
  59. $count = Database::fetch_row($result);
  60. $count = $count[0];
  61. //3. Calculating first, end, etc
  62. $total_pages = 0;
  63. if ($count > 0) {
  64. if (!empty($limit)) {
  65. $total_pages = ceil($count/$limit);
  66. }
  67. }
  68. if ($page > $total_pages) {
  69. $page = $total_pages;
  70. }
  71. $start = $limit * $page - $limit;
  72. if ($start < 0) {
  73. $start = 0;
  74. }
  75. $sql = "SELECT
  76. exe_id,
  77. exe_user_id,
  78. firstname,
  79. lastname,
  80. aa.status,
  81. start_date,
  82. exe_result,
  83. exe_weighting,
  84. exe_result/exe_weighting as score,
  85. exe_duration,
  86. questions_to_check,
  87. orig_lp_id
  88. FROM $user_table u
  89. INNER JOIN (
  90. SELECT
  91. t.exe_id,
  92. t.exe_user_id,
  93. status,
  94. start_date,
  95. exe_result,
  96. exe_weighting,
  97. exe_result/exe_weighting as score,
  98. exe_duration,
  99. questions_to_check,
  100. orig_lp_id
  101. FROM $track_exercise t
  102. LEFT JOIN $track_attempt a
  103. ON (a.exe_id = t.exe_id AND t.exe_user_id = a.user_id )
  104. WHERE t.status = 'incomplete' AND $where_condition
  105. GROUP BY exe_user_id
  106. ) as aa
  107. ON aa.exe_user_id = user_id
  108. ORDER BY $sidx $sord
  109. LIMIT $start, $limit";
  110. $result = Database::query($sql);
  111. $results = array();
  112. while ($row = Database::fetch_array($result, 'ASSOC')) {
  113. $results[] = $row;
  114. }
  115. $oExe = new exercise();
  116. $oExe->read($exercise_id);
  117. $response = new stdClass();
  118. $response->page = $page;
  119. $response->total = $total_pages;
  120. $response->records = $count;
  121. $i=0;
  122. if (!empty($results)) {
  123. foreach ($results as $row) {
  124. $sql = "SELECT SUM(count_question_id) as count_question_id
  125. FROM (
  126. SELECT 1 as count_question_id
  127. FROM $track_attempt a
  128. WHERE
  129. user_id = {$row['exe_user_id']} AND
  130. exe_id = {$row['exe_id']}
  131. GROUP by question_id
  132. ) as count_table";
  133. $result_count = Database::query($sql);
  134. $count_questions = Database::fetch_array($result_count,'ASSOC');
  135. $count_questions = $count_questions['count_question_id'];
  136. $row['count_questions'] = $count_questions;
  137. $response->rows[$i]['id'] = $row['exe_id'];
  138. $remaining = strtotime($row['start_date'])+($oExe->expired_time*60) - strtotime(api_get_utc_datetime(time()));
  139. $h = floor($remaining/3600);
  140. $m = floor(($remaining - ($h*3600))/60);
  141. $s = ($remaining - ($h*3600) - ($m*60));
  142. $array = array(
  143. $row['firstname'],
  144. $row['lastname'],
  145. api_format_date($row['start_date'], DATE_TIME_FORMAT_LONG).' ['.($h>0?$h.':':'').sprintf("%02d",$m).':'.sprintf("%02d",$s).']',
  146. $row['count_questions'],
  147. round($row['score']*100).'%'
  148. );
  149. $response->rows[$i]['cell'] = $array;
  150. $i++;
  151. }
  152. }
  153. echo json_encode($response);
  154. break;
  155. case 'update_exercise_list_order':
  156. if (api_is_allowed_to_edit(null, true)) {
  157. $new_list = $_REQUEST['exercise_list'];
  158. $table = Database::get_course_table(TABLE_QUIZ_ORDER);
  159. $counter = 1;
  160. //Drop all
  161. Database::query("DELETE FROM $table WHERE session_id = $session_id AND c_id = $course_id");
  162. //Insert all
  163. foreach ($new_list as $new_order_id) {
  164. Database::insert(
  165. $table,
  166. array(
  167. 'exercise_order' => $counter,
  168. 'session_id' => $session_id,
  169. 'exercise_id' => intval($new_order_id),
  170. 'c_id' => $course_id
  171. )
  172. );
  173. $counter++;
  174. }
  175. Display::display_confirmation_message(get_lang('Saved'));
  176. }
  177. break;
  178. case 'update_question_order':
  179. $course_info = api_get_course_info($course_code);
  180. $course_id = $course_info['real_id'];
  181. $exercise_id = isset($_REQUEST['exercise_id']) ? $_REQUEST['exercise_id'] : null;
  182. if (empty($exercise_id)) {
  183. return Display::display_error_message(get_lang('Error'));
  184. }
  185. if (api_is_allowed_to_edit(null, true)) {
  186. $new_question_list = $_POST['question_id_list'];
  187. $TBL_QUESTIONS = Database::get_course_table(TABLE_QUIZ_TEST_QUESTION);
  188. $counter = 1;
  189. foreach ($new_question_list as $new_order_id) {
  190. Database::update(
  191. $TBL_QUESTIONS,
  192. array('question_order' => $counter),
  193. array('question_id = ? AND c_id = ? AND exercice_id = ? ' => array(intval($new_order_id), $course_id, $exercise_id)))
  194. ;
  195. $counter++;
  196. }
  197. Display::display_confirmation_message(get_lang('Saved'));
  198. }
  199. break;
  200. case 'add_question_to_reminder':
  201. /** @var Exercise $objExercise */
  202. $objExercise = $_SESSION['objExercise'];
  203. if (empty($objExercise)) {
  204. echo 0;
  205. exit;
  206. } else {
  207. $objExercise->edit_question_to_remind(
  208. $_REQUEST['exe_id'],
  209. $_REQUEST['question_id'],
  210. $_REQUEST['action']
  211. );
  212. }
  213. break;
  214. case 'save_exercise_by_now':
  215. $course_info = api_get_course_info($course_code);
  216. $course_id = $course_info['real_id'];
  217. // Use have permissions?
  218. if (api_is_allowed_to_session_edit()) {
  219. // "all" or "simple" strings means that there's one or all questions exercise type
  220. $type = isset($_REQUEST['type']) ? $_REQUEST['type'] : null;
  221. // Questions choices.
  222. $choice = isset($_REQUEST['choice']) ? $_REQUEST['choice'] : null;
  223. // Hot spot coordinates from all questions.
  224. $hot_spot_coordinates = isset($_REQUEST['hotspot']) ? $_REQUEST['hotspot'] : null;
  225. // There is a reminder?
  226. $remind_list = isset($_REQUEST['remind_list']) && !empty($_REQUEST['remind_list']) ? array_keys($_REQUEST['remind_list']) : null;
  227. // Needed in manage_answer.
  228. $learnpath_id = isset($_REQUEST['learnpath_id']) ? intval($_REQUEST['learnpath_id']) : 0;
  229. $learnpath_item_id = isset($_REQUEST['learnpath_item_id']) ? intval($_REQUEST['learnpath_item_id']) : 0;
  230. // Attempt id.
  231. $exe_id = $_REQUEST['exe_id'];
  232. if ($debug) {
  233. error_log("exe_id = $exe_id");
  234. error_log("type = $type");
  235. error_log("choice = ".print_r($choice, 1)." ");
  236. error_log("hot_spot_coordinates = ".print_r($hot_spot_coordinates, 1));
  237. error_log("remind_list = ".print_r($remind_list, 1));
  238. }
  239. // Exercise information.
  240. /** @var Exercise $objExercise */
  241. $objExercise = isset($_SESSION['objExercise']) ? $_SESSION['objExercise'] : null;
  242. // Question info.
  243. $question_id = isset($_REQUEST['question_id']) ? intval($_REQUEST['question_id']) : null;
  244. $question_list = $_SESSION['questionList'];
  245. // If exercise or question is not set then exit.
  246. if (empty($question_list) || empty($objExercise)) {
  247. echo 'error';
  248. if ($debug) {
  249. if (empty($question_list)) {
  250. error_log("question_list is empty");
  251. }
  252. if (empty($objExercise)) {
  253. error_log("objExercise is empty");
  254. }
  255. }
  256. exit;
  257. }
  258. // Getting information of the current exercise.
  259. $exercise_stat_info = $objExercise->get_stat_track_exercise_info_by_exe_id($exe_id);
  260. $exercise_id = $exercise_stat_info['exe_exo_id'];
  261. $attempt_list = array();
  262. // First time here we create an attempt (getting the exe_id).
  263. if (empty($exercise_stat_info)) {
  264. } else {
  265. // We know the user we get the exe_id.
  266. $exe_id = $exercise_stat_info['exe_id'];
  267. $total_score = $exercise_stat_info['exe_result'];
  268. //Getting the list of attempts
  269. $attempt_list = get_all_exercise_event_by_exe_id($exe_id);
  270. }
  271. // Updating Reminder algorythm.
  272. if ($objExercise->type == ONE_PER_PAGE) {
  273. $bd_reminder_list = explode(',', $exercise_stat_info['questions_to_check']);
  274. if (empty($remind_list)) {
  275. $remind_list = $bd_reminder_list;
  276. $new_list = array();
  277. foreach ($bd_reminder_list as $item) {
  278. if ($item != $question_id) {
  279. $new_list[] = $item;
  280. }
  281. }
  282. $remind_list = $new_list;
  283. } else {
  284. if (isset($remind_list[0])) {
  285. if (!in_array($remind_list[0], $bd_reminder_list)) {
  286. array_push($bd_reminder_list, $remind_list[0]);
  287. }
  288. $remind_list = $bd_reminder_list;
  289. }
  290. }
  291. }
  292. // No exe id? Can't save answer.
  293. if (empty($exe_id)) {
  294. // Fires an error.
  295. echo 'error';
  296. if ($debug) {
  297. error_log("exe_id is empty");
  298. }
  299. exit;
  300. } else {
  301. $_SESSION['exe_id'] = $exe_id;
  302. }
  303. // Getting the total weight if the request is simple
  304. $total_weight = 0;
  305. if ($type == 'simple') {
  306. foreach ($question_list as $my_question_id) {
  307. $objQuestionTmp = Question::read($my_question_id, $course_id);
  308. $total_weight += $objQuestionTmp->selectWeighting();
  309. }
  310. }
  311. unset($objQuestionTmp);
  312. // Looping the question list
  313. foreach ($question_list as $my_question_id) {
  314. if ($debug) {
  315. error_log("Saving question_id = $my_question_id ");
  316. }
  317. if ($type == 'simple' && $question_id != $my_question_id) {
  318. continue;
  319. }
  320. $my_choice = isset($choice[$my_question_id]) ?
  321. $choice[$my_question_id] : null;
  322. if ($debug) {
  323. error_log("my_choice = ".print_r($my_choice, 1)."");
  324. }
  325. // Creates a temporary Question object
  326. $objQuestionTmp = Question::read($my_question_id, $course_id);
  327. // Getting free choice data.
  328. if ($objQuestionTmp->type == FREE_ANSWER && $type == 'all') {
  329. $my_choice = isset($_REQUEST['free_choice'][$my_question_id]) && !empty($_REQUEST['free_choice'][$my_question_id]) ? $_REQUEST['free_choice'][$my_question_id]: null;
  330. }
  331. if ($type == 'all') {
  332. $total_weight += $objQuestionTmp->selectWeighting();
  333. }
  334. // This variable came from exercise_submit_modal.php.
  335. $hotspot_delineation_result = null;
  336. if (isset($_SESSION['hotspot_delineation_result']) && isset($_SESSION['hotspot_delineation_result'][$objExercise->selectId()])) {
  337. $hotspot_delineation_result = $_SESSION['hotspot_delineation_result'][$objExercise->selectId()][$my_question_id];
  338. }
  339. if ($type == 'simple') {
  340. // Getting old attempt in order to decrees the total score.
  341. $old_result = $objExercise->manage_answer(
  342. $exe_id,
  343. $my_question_id,
  344. null,
  345. 'exercise_show',
  346. array(),
  347. false,
  348. true,
  349. false,
  350. $objExercise->selectPropagateNeg(),
  351. array()
  352. );
  353. // Removing old score.
  354. $total_score = $total_score - $old_result['score'];
  355. }
  356. // Deleting old attempt
  357. if (isset($attempt_list) && !empty($attempt_list[$my_question_id])) {
  358. if ($debug) {
  359. error_log("delete_attempt exe_id : $exe_id, my_question_id: $my_question_id");
  360. }
  361. delete_attempt($exe_id, api_get_user_id(), $course_code, $session_id, $my_question_id);
  362. if ($objQuestionTmp->type == HOT_SPOT) {
  363. delete_attempt_hotspot($exe_id, api_get_user_id(), $course_code, $session_id, $my_question_id);
  364. }
  365. if (isset($attempt_list[$my_question_id]) && isset($attempt_list[$my_question_id]['marks'])) {
  366. $total_score -= $attempt_list[$my_question_id]['marks'];
  367. }
  368. }
  369. // We're inside *one* question. Go through each possible answer for this question
  370. $result = $objExercise->manage_answer(
  371. $exe_id,
  372. $my_question_id,
  373. $my_choice,
  374. 'exercise_result',
  375. $hot_spot_coordinates,
  376. true,
  377. false,
  378. false,
  379. $objExercise->selectPropagateNeg(),
  380. $hotspot_delineation_result
  381. );
  382. // Adding the new score.
  383. $total_score += $result['score'];
  384. if ($debug) {
  385. error_log("total_score: $total_score ");
  386. error_log("total_weight: $total_weight ");
  387. }
  388. $duration = 0;
  389. $now = time();
  390. if ($type == 'all') {
  391. $exercise_stat_info = $objExercise->get_stat_track_exercise_info_by_exe_id($exe_id);
  392. }
  393. $key = get_time_control_key($exercise_id, $exercise_stat_info['orig_lp_id'], $exercise_stat_info['orig_lp_item_id']);
  394. if (isset($_SESSION['duration_time'][$key]) && !empty($_SESSION['duration_time'][$key])) {
  395. $duration = $now - $_SESSION['duration_time'][$key];
  396. if (!empty($exercise_stat_info['exe_duration'])) {
  397. $duration += $exercise_stat_info['exe_duration'];
  398. }
  399. $duration = intval($duration);
  400. } else {
  401. if (!empty($exercise_stat_info['exe_duration'])) {
  402. $duration = $exercise_stat_info['exe_duration'];
  403. }
  404. }
  405. $_SESSION['duration_time'][$key] = time();
  406. update_event_exercice(
  407. $exe_id,
  408. $objExercise->selectId(),
  409. $total_score,
  410. $total_weight,
  411. $session_id,
  412. $exercise_stat_info['orig_lp_id'],
  413. $exercise_stat_info['orig_lp_item_id'],
  414. $exercise_stat_info['orig_lp_item_view_id'],
  415. $duration,
  416. $question_list,
  417. 'incomplete',
  418. $remind_list
  419. );
  420. // Destruction of the Question object
  421. unset($objQuestionTmp);
  422. if ($debug) {
  423. error_log(" -- end question -- ");
  424. }
  425. }
  426. if ($debug) {
  427. error_log(" ------ end ajax call ------- ");
  428. }
  429. }
  430. if ($objExercise->type == ONE_PER_PAGE) {
  431. echo 'one_per_page';
  432. exit;
  433. }
  434. echo 'ok';
  435. break;
  436. default:
  437. echo '';
  438. }
  439. exit;