lp_ajax_switch_item.php 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309
  1. <?php
  2. /* For licensing terms, see /license.txt */
  3. use ChamiloSession as Session;
  4. /**
  5. * This script contains the server part of the xajax interaction process. The client part is located
  6. * in lp_api.php or other api's.
  7. * This is a first attempt at using xajax and AJAX in general, so the code might be a bit unsettling.
  8. *
  9. * @package chamilo.learnpath
  10. *
  11. * @author Yannick Warnier <ywarnier@beeznest.org>
  12. */
  13. // Flag to allow for anonymous user - needs to be set before global.inc.php
  14. $use_anonymous = true;
  15. require_once __DIR__.'/../inc/global.inc.php';
  16. /**
  17. * Get one item's details.
  18. *
  19. * @param int LP ID
  20. * @param int user ID
  21. * @param int View ID
  22. * @param int Current item ID
  23. * @param int New item ID
  24. */
  25. function switch_item_details($lp_id, $user_id, $view_id, $current_item, $next_item)
  26. {
  27. $debug = 0;
  28. $return = '';
  29. if ($debug > 0) {
  30. error_log(
  31. 'In xajax_switch_item_details('.$lp_id.','.$user_id.','.$view_id.','.$current_item.','.$next_item.')',
  32. 0
  33. );
  34. }
  35. //$objResponse = new xajaxResponse();
  36. /*$item_id may be one of:
  37. * -'next'
  38. * -'previous'
  39. * -'first'
  40. * -'last'
  41. * - a real item ID
  42. */
  43. $mylp = learnpath::getLpFromSession(api_get_course_id(), $lp_id, $user_id);
  44. $new_item_id = 0;
  45. switch ($next_item) {
  46. case 'next':
  47. $mylp->set_current_item($current_item);
  48. $mylp->next();
  49. $new_item_id = $mylp->get_current_item_id();
  50. if ($debug > 1) {
  51. error_log('In {next} - next item is '.$new_item_id.'(current: '.$current_item.')');
  52. }
  53. break;
  54. case 'previous':
  55. $mylp->set_current_item($current_item);
  56. $mylp->previous();
  57. $new_item_id = $mylp->get_current_item_id();
  58. if ($debug > 1) {
  59. error_log('In {previous} - next item is '.$new_item_id.'(current: '.$current_item.')');
  60. }
  61. break;
  62. case 'first':
  63. $mylp->set_current_item($current_item);
  64. $mylp->first();
  65. $new_item_id = $mylp->get_current_item_id();
  66. if ($debug > 1) {
  67. error_log('In {first} - next item is '.$new_item_id.'(current: '.$current_item.')');
  68. }
  69. break;
  70. case 'last':
  71. break;
  72. default:
  73. // Should be filtered to check it's not hacked.
  74. if ($next_item == $current_item) {
  75. // If we're opening the same item again.
  76. $mylp->items[$current_item]->restart();
  77. }
  78. $new_item_id = $next_item;
  79. $mylp->set_current_item($new_item_id);
  80. if ($debug > 1) {
  81. error_log('In {default} - next item is '.$new_item_id.'(current: '.$current_item.')');
  82. }
  83. break;
  84. }
  85. if (WhispeakAuthPlugin::isLpItemMarked($new_item_id)) {
  86. ChamiloSession::write(
  87. WhispeakAuthPlugin::SESSION_LP_ITEM,
  88. ['lp' => $lp_id, 'lp_item' => $new_item_id, 'src' => '']
  89. );
  90. }
  91. $mylp->start_current_item(true);
  92. if ($mylp->force_commit) {
  93. $mylp->save_current();
  94. }
  95. if (is_object($mylp->items[$new_item_id])) {
  96. $mylpi = $mylp->items[$new_item_id];
  97. } else {
  98. if ($debug > 1) {
  99. error_log('In switch_item_details - generating new item object', 0);
  100. }
  101. $mylpi = new learnpathItem($new_item_id, $user_id);
  102. $mylpi->set_lp_view($view_id);
  103. }
  104. /*
  105. * now get what's needed by the SCORM API:
  106. * -score
  107. * -max
  108. * -min
  109. * -lesson_status
  110. * -session_time
  111. * -suspend_data
  112. */
  113. $myscore = $mylpi->get_score();
  114. $mymax = $mylpi->get_max();
  115. if ($mymax === '') {
  116. $mymax = "''";
  117. }
  118. $mymin = $mylpi->get_min();
  119. $mylesson_status = $mylpi->get_status();
  120. $mylesson_location = $mylpi->get_lesson_location();
  121. $mytotal_time = $mylpi->get_scorm_time('js');
  122. $mymastery_score = $mylpi->get_mastery_score();
  123. $mymax_time_allowed = $mylpi->get_max_time_allowed();
  124. $mylaunch_data = $mylpi->get_launch_data();
  125. /*
  126. if ($mylpi->get_type() == 'asset') {
  127. // Temporary measure to save completion of an asset. Later on,
  128. // Chamilo should trigger something on unload, maybe...
  129. // (even though that would mean the last item cannot be completed)
  130. $mylesson_status = 'completed';
  131. $mylpi->set_status('completed');
  132. $mylpi->save();
  133. }
  134. */
  135. $mysession_time = $mylpi->get_total_time();
  136. $mysuspend_data = $mylpi->get_suspend_data();
  137. $mylesson_location = $mylpi->get_lesson_location();
  138. $myic = $mylpi->get_interactions_count();
  139. $myistring = '';
  140. for ($i = 0; $i < $myic; $i++) {
  141. $myistring .= ",[".$i.",'','','','','','','']";
  142. }
  143. if (!empty($myistring)) {
  144. $myistring = substr($myistring, 1);
  145. }
  146. /*
  147. * The following lines should reinitialize the values for the SCO
  148. * However, due to many complications, we are now relying more on the
  149. * LMSInitialize() call and its underlying lp_ajax_initialize.php call
  150. * so this code is technically deprecated (but the change of item_id should
  151. * remain). However, due to numerous technical issues with SCORM, we prefer
  152. * leaving it as a double-lock security. If removing, please test carefully
  153. * with both SCORM and proper learning path tracking.
  154. */
  155. $return .=
  156. "olms.score=".$myscore.";".
  157. "olms.max=".$mymax.";".
  158. "olms.min=".$mymin.";".
  159. "olms.lesson_status='".$mylesson_status."';".
  160. "olms.lesson_location='".$mylesson_location."';".
  161. "olms.session_time='".$mysession_time."';".
  162. "olms.suspend_data='".$mysuspend_data."';".
  163. "olms.total_time = '".$mytotal_time."';".
  164. "olms.mastery_score = '".$mymastery_score."';".
  165. "olms.max_time_allowed = '".$mymax_time_allowed."';".
  166. "olms.launch_data = '".$mylaunch_data."';".
  167. "olms.interactions = new Array(".$myistring.");".
  168. "olms.item_objectives = new Array();".
  169. "olms.G_lastError = 0;".
  170. "olms.G_LastErrorMessage = 'No error';".
  171. "olms.finishSignalReceived = 0;";
  172. /*
  173. * and re-initialise the rest
  174. * -lms_lp_id
  175. * -lms_item_id
  176. * -lms_old_item_id
  177. * -lms_new_item_id
  178. * -lms_initialized
  179. * -lms_progress_bar_mode
  180. * -lms_view_id
  181. * -lms_user_id
  182. */
  183. $mytotal = $mylp->getTotalItemsCountWithoutDirs();
  184. $mycomplete = $mylp->get_complete_items_count();
  185. $myprogress_mode = $mylp->get_progress_bar_mode();
  186. $myprogress_mode = ($myprogress_mode == '' ? '%' : $myprogress_mode);
  187. $mynext = $mylp->get_next_item_id();
  188. $myprevious = $mylp->get_previous_item_id();
  189. $myitemtype = $mylpi->get_type();
  190. $mylesson_mode = $mylpi->get_lesson_mode();
  191. $mycredit = $mylpi->get_credit();
  192. $mylaunch_data = $mylpi->get_launch_data();
  193. $myinteractions_count = $mylpi->get_interactions_count();
  194. //$myobjectives_count = $mylpi->get_objectives_count();
  195. $mycore_exit = $mylpi->get_core_exit();
  196. $return .=
  197. //"saved_lesson_status='not attempted';" .
  198. "olms.lms_lp_id=".$lp_id.";".
  199. "olms.lms_item_id=".$new_item_id.";".
  200. "olms.lms_old_item_id=0;".
  201. //"lms_been_synchronized=0;" .
  202. "olms.lms_initialized=0;".
  203. //"lms_total_lessons=".$mytotal.";" .
  204. //"lms_complete_lessons=".$mycomplete.";" .
  205. //"lms_progress_bar_mode='".$myprogress_mode."';" .
  206. "olms.lms_view_id=".$view_id.";".
  207. "olms.lms_user_id=".$user_id.";".
  208. "olms.next_item=".$new_item_id.";".// This one is very important to replace possible literal strings.
  209. "olms.lms_next_item=".$mynext.";".
  210. "olms.lms_previous_item=".$myprevious.";".
  211. "olms.lms_item_type = '".$myitemtype."';".
  212. "olms.lms_item_credit = '".$mycredit."';".
  213. "olms.lms_item_lesson_mode = '".$mylesson_mode."';".
  214. "olms.lms_item_launch_data = '".$mylaunch_data."';".
  215. "olms.lms_item_interactions_count = '".$myinteractions_count."';".
  216. "olms.lms_item_objectives_count = '".$myinteractions_count."';".
  217. "olms.lms_item_core_exit = '".$mycore_exit."';".
  218. "olms.asset_timer = 0;";
  219. $updateMinTime = '';
  220. if (Tracking::minimunTimeAvailable(api_get_session_id(), api_get_course_int_id())) {
  221. $timeLp = $mylp->getAccumulateWorkTime();
  222. $timeTotalCourse = $mylp->getAccumulateWorkTimeTotalCourse();
  223. // Minimum connection percentage
  224. $perc = 100;
  225. // Time from the course
  226. $tc = $timeTotalCourse;
  227. $sessionId = api_get_session_id();
  228. if (!empty($sessionId) && $sessionId != 0) {
  229. /*$sql = "SELECT hours, perc FROM plugin_licences_course_session WHERE session_id = $sessionId";
  230. $res = Database::query($sql);
  231. if (Database::num_rows($res) > 0) {
  232. $aux = Database::fetch_assoc($res);
  233. $perc = $aux['perc'];
  234. $tc = $aux['hours'] * 60;
  235. }*/
  236. }
  237. // Percentage of the learning paths
  238. $pl = 0;
  239. if (!empty($timeTotalCourse)) {
  240. $pl = $timeLp / $timeTotalCourse;
  241. }
  242. // Minimum time for each learning path
  243. $time_total = intval($pl * $tc * $perc / 100) * 60;
  244. //$time_total = $mylp->getAccumulateWorkTime() * 60;
  245. /*$lpTime = Tracking::get_time_spent_in_lp(
  246. $user_id,
  247. api_get_course_id(),
  248. [$lp_id],
  249. api_get_session_id()
  250. );*/
  251. $lpTimeList = Tracking::getCalculateTime($user_id, api_get_course_int_id(), api_get_session_id());
  252. $lpTime = isset($lpTimeList[TOOL_LEARNPATH][$lp_id]) ? $lpTimeList[TOOL_LEARNPATH][$lp_id] : 0;
  253. if ($lpTime >= $time_total) {
  254. $time_spent = $time_total;
  255. } else {
  256. $time_spent = $lpTime;
  257. }
  258. $hour = (intval($lpTime / 3600)) < 10 ? '0'.intval($lpTime / 3600) : intval($lpTime / 3600);
  259. $minute = date('i', $lpTime);
  260. $second = date('s', $lpTime);
  261. $updateMinTime = "update_time_bar('$time_spent','$time_total','%');".
  262. "update_chronometer('$hour','$minute','$second');";
  263. }
  264. $return .=
  265. "update_toc('unhighlight','".$current_item."');".
  266. "update_toc('highlight','".$new_item_id."');".
  267. "update_toc('$mylesson_status','".$new_item_id."');".
  268. "update_progress_bar('$mycomplete','$mytotal','$myprogress_mode');".
  269. $updateMinTime
  270. ;
  271. $return .= 'updateGamificationValues(); ';
  272. $mylp->set_error_msg('');
  273. $mylp->prerequisites_match(); // Check the prerequisites are all complete.
  274. if ($debug > 1) {
  275. error_log('Prereq_match() returned '.htmlentities($mylp->error), 0);
  276. }
  277. // Save the new item ID for the exercise tool to use.
  278. Session::write('scorm_item_id', $new_item_id);
  279. Session::write('lpobject', serialize($mylp));
  280. Session::write('oLP', $mylp);
  281. return $return;
  282. }
  283. echo switch_item_details(
  284. $_REQUEST['lid'],
  285. $_REQUEST['uid'],
  286. $_REQUEST['vid'],
  287. $_REQUEST['iid'],
  288. $_REQUEST['next']
  289. );