From 23fcfae9cbfeacd56376bbe3c1abe8a8405ddf21 Mon Sep 17 00:00:00 2001 From: Julio Montoya Date: Thu, 30 Dec 2010 16:45:30 +0100 Subject: [PATCH 1/9] Adding show_score function to use the new max and min platform scores --- main/exercice/exercice.php | 10 ++++++++-- main/exercice/exercise.lib.php | 24 +++++++++++++++++++++++- main/exercice/exercise_show.php | 9 +++++---- main/newscorm/lp_stats.php | 20 ++++++++++++-------- 4 files changed, 48 insertions(+), 15 deletions(-) diff --git a/main/exercice/exercice.php b/main/exercice/exercice.php index 97507676a2..907edde139 100755 --- a/main/exercice/exercice.php +++ b/main/exercice/exercice.php @@ -778,7 +778,13 @@ if ($show == 'test') { //Showing exercise title $row['title']=text_filter($row['title']); - echo Display::tag('h1',$row['title']); + echo Display::tag('h1',$row['title']); + + if ($session_id == $row['session_id']) { + //Settings + //echo Display::url(Display::return_icon('settings.png',get_lang('Edit'), array('width'=>'22px'))." ".get_lang('Edit'), 'exercise_admin.php?'.api_get_cidreq().'&modifyExercise=yes&exerciseId='.$row['id']); + } + echo '

'; echo $session_img; $exid = $row['id']; @@ -800,7 +806,7 @@ if ($show == 'test') { if ($session_id == $row['session_id']) { //Settings - echo Display::url(Display::return_icon('settings.png',get_lang('Edit'), array('width'=>'22px'))." ".get_lang('Edit'), 'exercise_admin.php?'.api_get_cidreq().'&modifyExercise=yes&exerciseId='.$row['id']); + echo Display::url(Display::return_icon('edit.gif',get_lang('Edit'), array('width'=>'20px')), 'exercise_admin.php?'.api_get_cidreq().'&modifyExercise=yes&exerciseId='.$row['id']); //Export echo Display::url(Display::return_icon('cd.gif', get_lang('CopyExercise')), '', array('onclick'=>"javascript:if(!confirm('".addslashes(api_htmlentities(get_lang('AreYouSureToCopy'),ENT_QUOTES,$charset))." ".addslashes($row['title'])."?"."')) return false;",'href'=>'exercice.php?'.api_get_cidreq().'&choice=copy_exercise&sec_token='.$token.'&exerciseId='.$row['id'])); diff --git a/main/exercice/exercise.lib.php b/main/exercice/exercise.lib.php index 188fec91d2..0be1607b12 100755 --- a/main/exercice/exercise.lib.php +++ b/main/exercice/exercise.lib.php @@ -1015,7 +1015,8 @@ function get_exam_results_data($from, $number_of_items, $column, $direction) { $ex = show_score($my_res, $my_total); - $result_list = round(($my_res / ($my_total != 0 ? $my_total : 1)) * 100, 2) . '% (' . $my_res . ' / ' . $my_total . ') --> '.$ex; + //$result_list = round(($my_res / ($my_total != 0 ? $my_total : 1)) * 100, 2) . '% (' . $my_res . ' / ' . $my_total . ')'; + $result_list = $ex; $html_link = ''; if ($is_allowedToEdit || $is_tutor) { @@ -1105,5 +1106,26 @@ function show_score($score, $weight, $show_porcentage = true) { } +function convert_score($score, $weight) { + $html = ''; + $score_rounded = $score; + + if ($score != '' && $weight != '') { + $max_note = api_get_setting('exercise_max_score'); + $min_note = api_get_setting('exercise_min_score'); + if ($max_note != '' && $min_note != '') { + + if (!empty($weight)) { + + $score = $min_note + ($max_note - $min_note) * $score /$weight; + } else { + $score = $min_note; + } + $score_rounded = round($score, 2); + } + } + return $score_rounded; +} + diff --git a/main/exercice/exercise_show.php b/main/exercice/exercise_show.php index 3672eb635c..6d8c04b47d 100755 --- a/main/exercice/exercise_show.php +++ b/main/exercice/exercise_show.php @@ -578,12 +578,13 @@ if ($show_results) { '; - //echo get_lang('Score')." : $my_total_score/$my_total_weight"; - echo get_lang('Score')." : ".show_score($my_total_score,$total_weighting,false); + echo get_lang('Score')." : $my_total_score/$my_total_weight"; + //echo get_lang('Score')." : ".show_score($my_total_score, $total_weighting, false); echo ''; unset($objAnswerTmp); diff --git a/main/newscorm/lp_stats.php b/main/newscorm/lp_stats.php index 57063c5dc3..0f5280d60b 100755 --- a/main/newscorm/lp_stats.php +++ b/main/newscorm/lp_stats.php @@ -14,6 +14,7 @@ require_once 'learnpath.class.php'; require_once 'resourcelinker.inc.php'; require_once api_get_path(LIBRARY_PATH).'tracking.lib.php'; require_once api_get_path(LIBRARY_PATH).'course.lib.php'; +require_once '../exercice/exercise.lib.php'; if (empty($_SESSION['_course']['id']) && isset($_GET['course'])) { $course_code = Security::remove_XSS($_GET['course']); @@ -285,7 +286,9 @@ if (is_array($list) && count($list) > 0) { if (!$is_allowed_to_edit && $result_disabled_ext_all) { $view_score = Display::return_icon('invisible.gif', get_lang('ResultsHiddenByExerciseSetting')); } else { - $view_score = ($score == 0 ? '/' : ($maxscore === 0 ? $score : $score . '/' . float_format($maxscore, 1))); + //$view_score = ($score == 0 ? '/' : ($maxscore === 0 ? $score : $score . '/' . float_format($maxscore, 1))); + $view_score = show_score($score,$maxscore, false); + } $output .= "\n" . "\n" . "$extend_attempt_link\n" . '' . get_lang('Attempt') . ' ' . $row['iv_view_count'] . "\n" . '

' . $my_lesson_status . "
\n" . '
' . $view_score . "
\n" . '
'.$time.'
'; @@ -296,7 +299,6 @@ if (is_array($list) && count($list) > 0) { $temp[] = Security::remove_XSS($my_lesson_status); if ($row['item_type'] == 'quiz') { - if (!$is_allowed_to_edit && $result_disabled_ext_all) { $temp[] = '/'; } else { @@ -546,13 +548,12 @@ if (is_array($list) && count($list) > 0) { $output .= "$extend_link\n" . '
' .$title. '
' . "\n"; $output .= '
' . $my_lesson_status . "
\n" . '
'; if ($row['item_type'] == 'quiz') { - if (!$is_allowed_to_edit && $result_disabled_ext_all) { $output .= Display::return_icon('invisible.gif', get_lang('ResultsHiddenByExerciseSetting')); } else { - $output .= ($score == 0 ? '0/'.float_format($maxscore, 1) : ($maxscore == 0 ? $score : float_format($score, 1) . '/' . float_format($maxscore, 1))); + // $output .= ($score == 0 ? '0/'.float_format($maxscore, 1) : ($maxscore == 0 ? $score : float_format($score, 1) . '/' . float_format($maxscore, 1))); + $output .= show_score($score, $maxscore, false); } - } else { $output .= ($score == 0 ? '/' : ($maxscore == 0 ? $score : $score . '/' . $maxscore)); } @@ -642,10 +643,13 @@ if (is_array($list) && count($list) > 0) { if ($my_score == 0 ) { $view_score = '0/'.$my_maxscore; } else { - if ($my_maxscore == 0) + if ($my_maxscore == 0) { $view_score = $my_score; - else - $view_score = $my_score . '/' . $my_maxscore; + } else { + //$view_score = $my_score . '/' . $my_maxscore; + $view_score = show_score($my_score, $my_maxscore, false); + } + } //$view_score = ($my_score == 0 ? '0.00/'.$my_maxscore : ($my_maxscore == 0 ? $my_score : $my_score . '/' . $my_maxscore)); } From 6f21b536c46d28e0bb4dade7e9667fde60e70f20 Mon Sep 17 00:00:00 2001 From: Julio Montoya Date: Fri, 31 Dec 2010 15:39:41 +0100 Subject: [PATCH 2/9] Adding function get_course_information_by_id, adding course id in function Coursemanager::get_course_list_of_user_as_course_admin() --- main/inc/lib/course.lib.php | 17 ++++++++++++++--- 1 file changed, 14 insertions(+), 3 deletions(-) diff --git a/main/inc/lib/course.lib.php b/main/inc/lib/course.lib.php index 827144538c..3377afbd31 100755 --- a/main/inc/lib/course.lib.php +++ b/main/inc/lib/course.lib.php @@ -133,6 +133,17 @@ class CourseManager { WHERE code='".Database::escape_string($course_code)."'"),'ASSOC' ); } + + /** + * Returns all the information of a given coursecode + * @param int the course id + * @return an array with all the fields of the course table + + */ + public static function get_course_information_by_id($course_id) { + return Database::select('*', Database::get_main_table(TABLE_MAIN_COURSE), array('id = ?' =>intval($course_id)) ); + } + /** * Returns a list of courses. Should work with quickform syntax @@ -612,7 +623,7 @@ class CourseManager { $user_id = intval($user_id); $data = array(); - $sql_nb_cours = "SELECT course_rel_user.course_code, course.title + $sql_nb_cours = "SELECT course_rel_user.course_code, course.title, course.id FROM $tbl_course_user as course_rel_user INNER JOIN $tbl_course as course ON course.code = course_rel_user.course_code @@ -623,7 +634,7 @@ class CourseManager { $tbl_course_rel_access_url = Database::get_main_table(TABLE_MAIN_ACCESS_URL_REL_COURSE); $access_url_id = api_get_current_access_url_id(); if ($access_url_id != -1) { - $sql_nb_cours = " SELECT course_rel_user.course_code, course.title + $sql_nb_cours = " SELECT course_rel_user.course_code, course.title, course.id FROM $tbl_course_user as course_rel_user INNER JOIN $tbl_course as course ON course.code = course_rel_user.course_code @@ -636,7 +647,7 @@ class CourseManager { $result_nb_cours = Database::query($sql_nb_cours); if (Database::num_rows($result_nb_cours) > 0) { - while ($row = Database::fetch_array($result_nb_cours)) { + while ($row = Database::fetch_array($result_nb_cours,'ASSOC')) { $data[$row['course_code']] = $row; } } From cb7364d0118365a2bbc592cc4750304f2921c599 Mon Sep 17 00:00:00 2001 From: Julio Montoya Date: Fri, 31 Dec 2010 15:40:11 +0100 Subject: [PATCH 3/9] Adding function SessionManager::get_sessions_by_coach --- main/inc/lib/sessionmanager.lib.php | 5 +++++ 1 file changed, 5 insertions(+) diff --git a/main/inc/lib/sessionmanager.lib.php b/main/inc/lib/sessionmanager.lib.php index 278902b9d6..18d7717d26 100755 --- a/main/inc/lib/sessionmanager.lib.php +++ b/main/inc/lib/sessionmanager.lib.php @@ -1231,4 +1231,9 @@ class SessionManager { return $return_array; } + public static function get_sessions_by_coach($user_id) { + $session_table = Database::get_main_table(TABLE_MAIN_SESSION); + return Database::select('*', $session_table, array('id_coach = ?'=>$user_id)); + } + } From cce4c6132fc15afe3bb4ab420af0c87ffbb86d85 Mon Sep 17 00:00:00 2001 From: Julio Montoya Date: Fri, 31 Dec 2010 15:40:59 +0100 Subject: [PATCH 4/9] More fixes for the new database classes --- main/inc/lib/database.lib.php | 81 +++++++++++++++++++++-------------- 1 file changed, 49 insertions(+), 32 deletions(-) diff --git a/main/inc/lib/database.lib.php b/main/inc/lib/database.lib.php index 81a24bbee7..ad5695ace6 100755 --- a/main/inc/lib/database.lib.php +++ b/main/inc/lib/database.lib.php @@ -1328,8 +1328,8 @@ class Database { * @todo lot of stuff to do here */ - public static function find($table_name, $columns = '*' , $where_conditions = array(), $option = 'ASSOC') { - $where_return = self::parse_where_conditions($where_conditions); + public static function select($columns = '*' , $table_name, $conditions = array(), $option = 'ASSOC') { + $conditions = self::parse_conditions($conditions); $clean_columns = ''; if (is_array($columns)) { $clean_columns = implode(',', $columns); @@ -1339,54 +1339,71 @@ class Database { } else { $clean_columns = (string)$columns; } - } - $sql = "SELECT $clean_columns FROM $table_name $where_return "; - $result = self::query($sql); - - $array = array(); - if ($result !== false) { // For isolation from database engine's behaviour. + } + + $sql = "SELECT $clean_columns FROM $table_name $conditions"; + $result = self::query($sql); + $array = array(); + if (self::num_rows($result) > 1 ) { while ($row = self::fetch_array($result, $option)) { if (isset($row['id'])) { $array[$row['id']] = $row; } else { $array[] = $row; } - } + } + } else { + $array = self::fetch_array($result, $option); } return $array; } /** - * Parses where conditionsof this form: array('id = ?' =>'4') + * Parses WHERE/ORDER conditions i.e array('where'=>array('id = ?' =>'4'), 'order'=>'id DESC')) + * @param array * @todo lot of stuff to do here */ - private function parse_where_conditions($conditions) { + private function parse_conditions($conditions) { if (empty($conditions)) { return ''; } - $where_return = ''; - foreach ($conditions as $condition => $value_array) { - if (is_array($value_array)) { - $clean_values = array(); - foreach($value_array as $item) { - $item = Database::escape_string($item); - $clean_values[]= "'$item'"; - } - } else { - $value_array = Database::escape_string($value_array); - $clean_values = "'$value_array'"; - } - if (!empty($condition) && !empty($clean_values)) { - $condition = str_replace('?','%s', $condition); //we treat everything as string - $condition = vsprintf($condition, $clean_values); - $where_return .= $condition; - } - } - if (!empty($where_return)) { - $where_return = " WHERE $where_return "; + $return_value = ''; + foreach ($conditions as $type_condition => $condition_data) { + switch($type_condition) { + case 'where': + foreach ($condition_data as $condition => $value_array) { + if (is_array($value_array)) { + $clean_values = array(); + foreach($value_array as $item) { + $item = Database::escape_string($item); + $clean_values[]= "'$item'"; + } + } else { + $value_array = Database::escape_string($value_array); + $clean_values = "'$value_array'"; + } + if (!empty($condition) && !empty($clean_values)) { + $condition = str_replace('?','%s', $condition); //we treat everything as string + $condition = vsprintf($condition, $clean_values); + $where_return .= $condition; + } + } + if (!empty($where_return)) { + $return_value = " WHERE $where_return" ; + } + break; + case 'order': + $return_value .= " ORDER BY $condition_data"; + break; + + } } - return $where_return; + return $return_value; + } + + private function parse_where_conditions($coditions){ + return self::parse_conditions(array('where'=>$coditions)); } /** From 2281d534ae5a3d79fe2c993ea3a672dc3a37eb65 Mon Sep 17 00:00:00 2001 From: Julio Montoya Date: Fri, 31 Dec 2010 15:51:02 +0100 Subject: [PATCH 5/9] Now we can copy a question from other course see BT#1917 --- main/exercice/answer.class.php | 96 ++++++++++--- main/exercice/exercise.class.php | 33 +++-- main/exercice/exercise.lib.php | 11 +- main/exercice/question.class.php | 141 +++++++++++------- main/exercice/question_list_admin.inc.php | 9 +- main/exercice/question_pool.php | 166 ++++++++++++++-------- 6 files changed, 301 insertions(+), 155 deletions(-) diff --git a/main/exercice/answer.class.php b/main/exercice/answer.class.php index 9cf38b6ebc..1d0f755aef 100755 --- a/main/exercice/answer.class.php +++ b/main/exercice/answer.class.php @@ -38,6 +38,7 @@ class Answer public $nbrAnswers; public $new_nbrAnswers; public $new_destination; // id of the next question if feedback option is set to Directfeedback + public $course; /** * constructor of the class @@ -45,7 +46,7 @@ class Answer * @author Olivier Brouckaert * @param integer Question ID that answers belong to */ - function Answer($questionId) { + function Answer($questionId, $course_id = null) { //$this->questionType=$questionType; $this->questionId = intval($questionId); $this->answer = array(); @@ -58,11 +59,20 @@ class Answer $this->destination = array(); // clears $new_* arrays $this->cancel(); + + if (!empty($course_id)) { + $this->course_id = intval($course_id); + $course_info = api_get_course_info_by_id($this->course_id); + } else { + $course_info = api_get_course_info(); + } + $this->course = $course_info; + // fills arrays - $objExercise = new Exercise(); + $objExercise = new Exercise($this->course['real_id']); $objExercise->read($_REQUEST['exerciseId']); - if($objExercise->random_answers=='1') { + if ($objExercise->random_answers=='1') { $this->readOrderedBy('rand()', '');// randomize answers } else { $this->read(); // natural order @@ -91,9 +101,8 @@ class Answer * * @author - Olivier Brouckaert */ - function read() { - global $_course; - $TBL_ANSWER = Database::get_course_table(TABLE_QUIZ_ANSWER); + function read() { + $TBL_ANSWER = Database::get_course_table(TABLE_QUIZ_ANSWER, $this->course['db_name']); $questionId=$this->questionId; //$answerType=$this->selectType(); @@ -127,8 +136,7 @@ class Answer * @param string DESC or ASC * @author Frederic Vauthier */ - function readOrderedBy($field,$order='ASC') { - global $_course; + function readOrderedBy($field,$order='ASC') { $field = Database::escape_string($field); if (empty($field)) { $field = 'position'; @@ -137,7 +145,7 @@ class Answer if ($order != 'ASC' && $order!='DESC') { $order = 'ASC'; } - $TBL_ANSWER = Database::get_course_table(TABLE_QUIZ_ANSWER); + $TBL_ANSWER = Database::get_course_table(TABLE_QUIZ_ANSWER, $this->course['db_name']); $questionId=$this->questionId; $sql="SELECT answer,correct,comment,ponderation,position, hotspot_coordinates, hotspot_type, destination, id_auto " . "FROM $TBL_ANSWER WHERE question_id='".$questionId."' " . @@ -223,7 +231,7 @@ class Answer * return array answer by id else return a bool */ function selectAnswerByAutoId($auto_id) { - $TBL_ANSWER = Database::get_course_table(TABLE_QUIZ_ANSWER); + $TBL_ANSWER = Database::get_course_table(TABLE_QUIZ_ANSWER, $this->course['db_name']); $auto_id = intval($auto_id); $sql="SELECT id, answer FROM $TBL_ANSWER WHERE id_auto='$auto_id'"; $rs = Database::query($sql); @@ -304,7 +312,7 @@ class Answer */ function getQuestionType() { - $TBL_QUESTIONS = Database::get_course_table(TABLE_QUIZ_QUESTION); + $TBL_QUESTIONS = Database::get_course_table(TABLE_QUIZ_QUESTION, $this->course['db_name']); $sql = "SELECT type FROM $TBL_QUESTIONS WHERE id = '".$this->questionId."'"; $res = Database::query($sql); if(Database::num_rows($res)<=0){ @@ -424,7 +432,7 @@ class Answer */ function updateAnswers($answer,$comment,$weighting,$position,$destination) { - $TBL_REPONSES = Database :: get_course_table(TABLE_QUIZ_ANSWER); + $TBL_REPONSES = Database :: get_course_table(TABLE_QUIZ_ANSWER, $this->course['db_name']); $questionId=$this->questionId; $sql = "UPDATE $TBL_REPONSES SET " . @@ -446,7 +454,7 @@ class Answer */ function save() { - $TBL_REPONSES = Database :: get_course_table(TABLE_QUIZ_ANSWER); + $TBL_REPONSES = Database :: get_course_table(TABLE_QUIZ_ANSWER, $this->course['db_name']); $questionId=$this->questionId; @@ -494,23 +502,65 @@ class Answer /** * Duplicates answers by copying them into another question * - * @author - Olivier Brouckaert - * @param - integer $newQuestionId - ID of the new question + * @author Olivier Brouckaert + * @param int $newQuestionId - ID of the new question */ - function duplicate($newQuestionId) - { - $TBL_REPONSES = Database :: get_course_table(TABLE_QUIZ_ANSWER); + function duplicate($newQuestionId, $course_info = null) { + require_once api_get_path(LIBRARY_PATH).'document.lib.php'; + + if (empty($course_info)) { + $course_info = $this->course; + } else { + $course_info = $course_info; + } + + $TBL_REPONSES = Database :: get_course_table(TABLE_QUIZ_ANSWER, $course_info['db_name']); + + if (self::getQuestionType() == MULTIPLE_ANSWER_TRUE_FALSE) { + + var_dump($this->selectQuestionId(), $newQuestionId); + + //Selecting origin options + + $origin_options = Question::readQuestionOption($this->selectQuestionId(),$this->course['db_name']); + var_dump($origin_options); + if (!empty($origin_options)) { + foreach($origin_options as $item) { + $new_option_list[]=$item['id']; + } + } + + + $destination_options = Question::readQuestionOption($newQuestionId,$course_info['db_name']); + $i=0; + $fixed_list = array(); + if (!empty($destination_options)) { + foreach($destination_options as $item) { + $fixed_list[$new_option_list[$i]] = $item['id']; + $i++; + } + } + var_dump($fixed_list); + } // if at least one answer - if($this->nbrAnswers) { + if ($this->nbrAnswers) { // inserts new answers into data base - $sql="INSERT INTO $TBL_REPONSES" . - "(id,question_id,answer,correct,comment," . - "ponderation,position,hotspot_coordinates,hotspot_type,destination) VALUES"; - + $sql="INSERT INTO $TBL_REPONSES (id,question_id,answer,correct,comment, ponderation,position,hotspot_coordinates,hotspot_type,destination) VALUES"; for($i=1;$i <= $this->nbrAnswers;$i++) { + if ($course_info['db_name'] != $this->course['db_name']) { + $this->answer[$i] = DocumentManager::replace_urls_inside_content_html_from_copy_course($this->answer[$i],$this->course['id'], $course_info['id']) ; + $this->comment[$i] = DocumentManager::replace_urls_inside_content_html_from_copy_course($this->comment[$i],$this->course['id'], $course_info['id']) ; + } + $answer = Database::escape_string($this->answer[$i]); $correct = Database::escape_string($this->correct[$i]); + + if (self::getQuestionType() == MULTIPLE_ANSWER_TRUE_FALSE) { + var_dump($correct); + $correct = $fixed_list[intval($correct)]; + } + $comment = Database::escape_string($this->comment[$i]); $weighting = Database::escape_string($this->weighting[$i]); $position = Database::escape_string($this->position[$i]); diff --git a/main/exercice/exercise.class.php b/main/exercice/exercise.class.php index 5a3d940a5f..de067b34c9 100755 --- a/main/exercice/exercise.class.php +++ b/main/exercice/exercise.class.php @@ -37,14 +37,16 @@ class Exercise { public $start_time; public $questionList; // array with the list of this exercise's questions public $results_disabled; - public $expired_time; + public $expired_time; + public $course; + /** * Constructor of the class * * @author - Olivier Brouckaert */ - function Exercise() { + function Exercise($course_id = null) { $this->id = 0; $this->exercise = ''; $this->description = ''; @@ -59,6 +61,15 @@ class Exercise { $this->start_time = '0000-00-00 00:00:00'; $this->results_disabled = 1; $this->expired_time = '0000-00-00 00:00:00'; + + if (!empty($course_id)) { + $this->course_id = intval($course_id); + $course_info = api_get_course_info_by_id($this->course_id); + } else { + $course_info = api_get_course_info(); + } + $this->course = $course_info; + } /** @@ -69,9 +80,9 @@ class Exercise { * @return - boolean - true if exercise exists, otherwise false */ function read($id) { - $TBL_EXERCICE_QUESTION = Database::get_course_table(TABLE_QUIZ_TEST_QUESTION); - $TBL_EXERCICES = Database::get_course_table(TABLE_QUIZ_TEST); - $TBL_QUESTIONS = Database::get_course_table(TABLE_QUIZ_QUESTION); + $TBL_EXERCICE_QUESTION = Database::get_course_table(TABLE_QUIZ_TEST_QUESTION,$this->course['db_name']); + $TBL_EXERCICES = Database::get_course_table(TABLE_QUIZ_TEST,$this->course['db_name']); + $TBL_QUESTIONS = Database::get_course_table(TABLE_QUIZ_QUESTION,$this->course['db_name']); #$TBL_REPONSES = Database::get_course_table(TABLE_QUIZ_ANSWER); $sql="SELECT title,description,sound,type,random, random_answers, active, results_disabled, max_attempt,start_time,end_time,feedback_type,expired_time FROM $TBL_EXERCICES WHERE id='".Database::escape_string($id)."'"; @@ -377,9 +388,9 @@ class Exercise { * @param - string $delete - ask to delete the file */ function updateSound($sound,$delete) { - global $audioPath, $documentPath,$_course, $_user; - $TBL_DOCUMENT = Database::get_course_table(TABLE_DOCUMENT); - $TBL_ITEM_PROPERTY = Database::get_course_table(TABLE_ITEM_PROPERTY); + global $audioPath, $documentPath; + $TBL_DOCUMENT = Database::get_course_table(TABLE_DOCUMENT, $this->course['db_name']); + $TBL_ITEM_PROPERTY = Database::get_course_table(TABLE_ITEM_PROPERTY,$this->course['db_name']); if ($sound['size'] && (strstr($sound['type'],'audio') || strstr($sound['type'],'video'))) { $this->sound=$sound['name']; @@ -393,7 +404,7 @@ class Exercise { /*$query="INSERT INTO $TBL_DOCUMENT(path,filetype) VALUES " ." ('".str_replace($documentPath,'',$audioPath).'/'.$this->sound."','file')"; Database::query($query);*/ - $id = add_document($_course,str_replace($documentPath,'',$audioPath).'/'.$this->sound,'file',$sound['size'],$sound['name']); + $id = add_document($this->course,str_replace($documentPath,'',$audioPath).'/'.$this->sound,'file',$sound['size'],$sound['name']); //$id = Database::insert_id(); //$time = time(); @@ -404,8 +415,8 @@ class Exercise { ." VALUES " ."('".TOOL_DOCUMENT."', $id, $_user['user_id'], 0, '$time', '$time', 'DocumentAdded' )"; Database::query($query);*/ - api_item_property_update($_course, TOOL_DOCUMENT, $id, 'DocumentAdded',$_user['user_id']); - item_property_update_on_folder($_course,str_replace($documentPath,'',$audioPath),$_user['user_id']); + api_item_property_update($this->course, TOOL_DOCUMENT, $id, 'DocumentAdded',api_get_user_id()); + item_property_update_on_folder($this->course,str_replace($documentPath,'',$audioPath),api_get_user_id()); } } } elseif($delete && is_file($audioPath.'/'.$this->sound)) { diff --git a/main/exercice/exercise.lib.php b/main/exercice/exercise.lib.php index 0be1607b12..a8f02cb9fb 100755 --- a/main/exercice/exercise.lib.php +++ b/main/exercice/exercise.lib.php @@ -92,6 +92,7 @@ function showQuestion($questionId, $onlyAnswers = false, $origin = false, $curre $nbrAnswers=$objAnswerTmp->selectNbrAnswers(); $quiz_question_options = Question::readQuestionOption($questionId); + // For "matching" type here, we need something a little bit special // because the match between the suggestions and the answers cannot be @@ -301,10 +302,12 @@ function showQuestion($questionId, $onlyAnswers = false, $origin = false, $curre } elseif ($answerType == MULTIPLE_ANSWER_TRUE_FALSE) { $options = array('type'=>'radio','name'=>'choice['.$questionId.']['.$numAnswer.']', 'class'=>'checkbox'); $s .=''; - $s .= Display::tag('td', $answer); - foreach ($quiz_question_options as $id=>$item) { - $options['value'] = $id; - $s .= Display::tag('td', Display::tag('input','',$options )); + $s .= Display::tag('td', $answer); + if (!empty($quiz_question_options)) { + foreach ($quiz_question_options as $id=>$item) { + $options['value'] = $id; + $s .= Display::tag('td', Display::tag('input','',$options )); + } } $s.=''; } diff --git a/main/exercice/question.class.php b/main/exercice/question.class.php index 8739fb3f4b..8a121b399b 100755 --- a/main/exercice/question.class.php +++ b/main/exercice/question.class.php @@ -46,7 +46,8 @@ abstract class Question public $picture; public $exerciseList; // array with the list of exercises which this question is in private $isContent; - + public $course; + static $typePicture = 'new_question.png'; static $explanationLangVar = ''; static $questionTypes = array( @@ -92,20 +93,25 @@ abstract class Question * @param - integer $id - question ID * @return - boolean - true if question exists, otherwise false */ - static function read($id) - { - global $_course; - - $TBL_EXERCICES = Database::get_course_table(TABLE_QUIZ_TEST); - $TBL_QUESTIONS = Database::get_course_table(TABLE_QUIZ_QUESTION); - $TBL_EXERCICE_QUESTION = Database::get_course_table(TABLE_QUIZ_TEST_QUESTION); - - $sql="SELECT question,description,ponderation,position,type,picture,level,extra FROM $TBL_QUESTIONS WHERE id='".Database::escape_string($id)."'"; + static function read($id, $course_id = null) { + + if (!empty($course_id)) { + $course_info = api_get_course_info_by_id($course_id); + } else { + global $course; + $course_info = api_get_course_info(); + } + + $TBL_EXERCICES = Database::get_course_table(TABLE_QUIZ_TEST, $course_info['db_name']); + $TBL_QUESTIONS = Database::get_course_table(TABLE_QUIZ_QUESTION, $course_info['db_name']); + $TBL_EXERCICE_QUESTION = Database::get_course_table(TABLE_QUIZ_TEST_QUESTION, $course_info['db_name']); + $id = intval($id); + $sql="SELECT question,description,ponderation,position,type,picture,level,extra FROM $TBL_QUESTIONS WHERE id='".$id."'"; $result=Database::query($sql); // if the question has been found - if($object=Database::fetch_object($result)) { + if ($object=Database::fetch_object($result)) { $objQuestion = Question::getInstance($object->type); $objQuestion->id = $id; $objQuestion->question = $object->question; @@ -116,8 +122,9 @@ abstract class Question $objQuestion->picture = $object->picture; $objQuestion->level = (int) $object->level; $objQuestion->extra = $object->extra; + $objQuestion->course = $course_info; - $sql="SELECT exercice_id FROM $TBL_EXERCICE_QUESTION WHERE question_id='".intval($id)."'"; + $sql="SELECT exercice_id FROM $TBL_EXERCICE_QUESTION WHERE question_id='".$id."'"; $result=Database::query($sql); // fills the array with the exercises which this question is in @@ -218,13 +225,11 @@ abstract class Question return $this->picture; } - function selectPicturePath() { - global $_course; + function selectPicturePath() { if (!empty($this->picture)) { - return api_get_path(WEB_COURSE_PATH).$_course['path'].'/document/images/'.$this->picture; + return api_get_path(WEB_COURSE_PATH).$this->course['path'].'/document/images/'.$this->picture; } - return false; - + return false; } /** @@ -312,7 +317,7 @@ abstract class Question * @param - integer $type - answer type */ function updateType($type) { - global $TBL_REPONSES; + $TBL_REPONSES = Database::get_course_table(TABLE_QUIZ_ANSWER, $this->course['db_name']); // if we really change the type if($type != $this->type) { @@ -336,16 +341,16 @@ abstract class Question * @return - boolean - true if uploaded, otherwise false */ function uploadPicture($Picture,$PictureName) { - global $picturePath, $_course, $_user; + global $picturePath; if (!file_exists($picturePath)) { if (mkdir($picturePath, api_get_permissions_for_new_directories())) { // document path - $documentPath = api_get_path(SYS_COURSE_PATH) . $_course['path'] . "/document"; + $documentPath = api_get_path(SYS_COURSE_PATH) . $this->course['path'] . "/document"; $path = str_replace($documentPath,'',$picturePath); $title_path = basename($picturePath); - $doc_id = add_document($_course, $path, 'folder', 0,$title_path); - api_item_property_update($_course, TOOL_DOCUMENT, $doc_id, 'FolderCreated', $_user['user_id']); + $doc_id = add_document($this->course, $path, 'folder', 0,$title_path); + api_item_property_update($this->course, TOOL_DOCUMENT, $doc_id, 'FolderCreated', api_get_user_id()); } } @@ -356,15 +361,15 @@ abstract class Question if($extension == 'gif' || $extension == 'png') { $o_img = new image($Picture); $o_img->send_image('JPG',$picturePath.'/'.$this->picture); - $document_id = add_document($_course, '/images/'.$this->picture, 'file', filesize($picturePath.'/'.$this->picture),$this->picture); + $document_id = add_document($this->course, '/images/'.$this->picture, 'file', filesize($picturePath.'/'.$this->picture),$this->picture); } else { move_uploaded_file($Picture,$picturePath.'/'.$this->picture)?true:false; } - $document_id = add_document($_course, '/images/'.$this->picture, 'file', filesize($picturePath.'/'.$this->picture),$this->picture); + $document_id = add_document($this->course, '/images/'.$this->picture, 'file', filesize($picturePath.'/'.$this->picture),$this->picture); if($document_id) { - return api_item_property_update($_course, TOOL_DOCUMENT, $document_id, 'DocumentAdded', $_user['user_id']); + return api_item_property_update($this->course, TOOL_DOCUMENT, $document_id, 'DocumentAdded', api_get_user_id); } } @@ -589,10 +594,9 @@ abstract class Question * @param - integer $exerciseId - exercise ID if saving in an exercise */ function save($exerciseId=0) { - global $_course,$_user; - $TBL_EXERCICE_QUESTION = Database::get_course_table(TABLE_QUIZ_TEST_QUESTION); - $TBL_QUESTIONS = Database::get_course_table(TABLE_QUIZ_QUESTION); + $TBL_EXERCICE_QUESTION = Database::get_course_table(TABLE_QUIZ_TEST_QUESTION, $this->course['db_name']); + $TBL_QUESTIONS = Database::get_course_table(TABLE_QUIZ_QUESTION, $this->course['db_name']); $id=$this->id; $question=$this->question; @@ -618,7 +622,7 @@ abstract class Question WHERE id='".Database::escape_string($id)."'"; Database::query($sql); if(!empty($exerciseId)) { - api_item_property_update($_course, TOOL_QUIZ, $id,'QuizQuestionUpdated',$_user['user_id']); + api_item_property_update($this->course, TOOL_QUIZ, $id,'QuizQuestionUpdated',api_get_user_id); } if (api_get_setting('search_enabled')=='true') { if ($exerciseId != 0) { @@ -653,12 +657,12 @@ abstract class Question $this->id=Database::insert_id(); - api_item_property_update($_course, TOOL_QUIZ, $this->id,'QuizQuestionAdded',$_user['user_id']); + api_item_property_update($this->course, TOOL_QUIZ, $this->id,'QuizQuestionAdded',api_get_user_id()); // If hotspot, create first answer if ($type == HOT_SPOT || $type == HOT_SPOT_ORDER) { $TBL_ANSWERS = Database::get_course_table(TABLE_QUIZ_ANSWER); - $sql="INSERT INTO $TBL_ANSWERS (`id` , `question_id` , `answer` , `correct` , `comment` , `ponderation` , `position` , `hotspot_coordinates` , `hotspot_type` ) VALUES ('1', '".Database::escape_string($this->id)."', '', NULL , '', '10' , '1', '0;0|0|0', 'square')"; + $sql="INSERT INTO $TBL_ANSWERS (id , question_id , answer , correct , comment , ponderation , position , hotspot_coordinates , hotspot_type ) VALUES ('1', '".Database::escape_string($this->id)."', '', NULL , '', '10' , '1', '0;0|0|0', 'square')"; Database::query($sql); } @@ -802,7 +806,7 @@ abstract class Question * @param - boolean $fromSave - comming from $this->save() or not */ function addToList($exerciseId, $fromSave = FALSE) { - $TBL_EXERCICE_QUESTION = Database::get_course_table(TABLE_QUIZ_TEST_QUESTION); + $TBL_EXERCICE_QUESTION = Database::get_course_table(TABLE_QUIZ_TEST_QUESTION, $this->course['db_name']); $id = $this->id; // checks if the exercise ID is not in the list if (!in_array($exerciseId,$this->exerciseList)) { @@ -861,7 +865,7 @@ abstract class Question } /** - * deletes a question from the database + * Deletes a question from the database * the parameter tells if the question is removed from all exercises (value = 0), * or just from one exercise (value = exercise ID) * @@ -871,9 +875,9 @@ abstract class Question function delete($deleteFromEx=0) { global $_course,$_user; - $TBL_EXERCICE_QUESTION = Database::get_course_table(TABLE_QUIZ_TEST_QUESTION); - $TBL_QUESTIONS = Database::get_course_table(TABLE_QUIZ_QUESTION); - $TBL_REPONSES = Database::get_course_table(TABLE_QUIZ_ANSWER); + $TBL_EXERCICE_QUESTION = Database::get_course_table(TABLE_QUIZ_TEST_QUESTION, $this->course['db_name']); + $TBL_QUESTIONS = Database::get_course_table(TABLE_QUIZ_QUESTION, $this->course['db_name']); + $TBL_REPONSES = Database::get_course_table(TABLE_QUIZ_ANSWER, $this->course['db_name']); $id=$this->id; @@ -900,7 +904,7 @@ abstract class Question $sql="DELETE FROM $TBL_REPONSES WHERE question_id='".Database::escape_string($id)."'"; Database::query($sql); - api_item_property_update($_course, TOOL_QUIZ, $id,'QuizQuestionDeleted',$_user['user_id']); + api_item_property_update($this->course, TOOL_QUIZ, $id,'QuizQuestionDeleted',api_get_user_id()); $this->removePicture(); // resets the object @@ -914,7 +918,7 @@ abstract class Question // disassociate question with this exercise $this -> search_engine_edit($deleteFromEx, FALSE, TRUE); } - api_item_property_update($_course, TOOL_QUIZ, $id,'QuizQuestionDeleted',$_user['user_id']); + api_item_property_update($this->course, TOOL_QUIZ, $id,'QuizQuestionDeleted',api_get_user_id()); } } @@ -923,9 +927,16 @@ abstract class Question * * @author - Olivier Brouckaert * @return - integer - ID of the new question - */ - function duplicate() { - global $TBL_QUESTIONS, $picturePath; + */ + + function duplicate($course_info = null) { + if (empty($course_info)) { + $course_info = $this->course; + } else { + $course_info = $course_info; + } + $TBL_QUESTIONS = Database::get_course_table(TABLE_QUIZ_QUESTION, $course_info['db_name']); + $TBL_QUESTION_OPTIONS = Database::get_course_table(TABLE_QUIZ_QUESTION_OPTION, $course_info['db_name']); $question = $this->question; $description = $this->description; @@ -933,15 +944,34 @@ abstract class Question $position = $this->position; $type = $this->type; $level = intval($this->level); - - $sql="INSERT INTO $TBL_QUESTIONS(question, description, ponderation, position, type, level ) VALUES('".Database::escape_string($question)."','".Database::escape_string($description)."','".Database::escape_string($weighting)."','".Database::escape_string($position)."','".Database::escape_string($type)."' ,'".Database::escape_string($level)."')"; - Database::query($sql); - - $id=Database::insert_id(); + $extra = $this->extra; + + require_once api_get_path(LIBRARY_PATH).'document.lib.php'; + if ($course_info['db_name'] != $this->course['db_name']) { + $description = DocumentManager::replace_urls_inside_content_html_from_copy_course($description, $this->course['id'], $course_info['id']); + $question = DocumentManager::replace_urls_inside_content_html_from_copy_course($question, $this->course['id'], $course_info['id']); + } + + $options = self::readQuestionOption($this->id); + var_dump($options); + + $sql="INSERT INTO $TBL_QUESTIONS(question, description, ponderation, position, type, level, extra ) VALUES('".Database::escape_string($question)."','".Database::escape_string($description)."','".Database::escape_string($weighting)."','".Database::escape_string($position)."','".Database::escape_string($type)."' ,'".Database::escape_string($level)."' ,'".Database::escape_string($extra)."' )"; + Database::query($sql); + + $new_question_id =Database::insert_id(); + + if (!empty($options)) { + //Saving the quiz_options + foreach ($options as $item) { + $item['question_id'] = $new_question_id; + unset($item['id']); + Database::insert($TBL_QUESTION_OPTIONS,$item); + } + } + // duplicates the picture - $this->exportPicture($id); - - return $id; + $this->exportPicture($new_question_id); + return $new_question_id; } /** @@ -967,8 +997,7 @@ abstract class Question * A subclass can redifine this function to add fields... * @param FormValidator $form the formvalidator instance (by reference) */ - function createForm (&$form,$fck_config=0) - { + function createForm (&$form,$fck_config=0) { echo '