2 /************************************************************************/
4 /************************************************************************/
5 /* Copyright (c) 2002-2008 by Greg Gay, Joel Kronenberg & Heidi Hazelton*/
6 /* Adaptive Technology Resource Centre / University of Toronto */
9 /* This program is free software. You can redistribute it and/or */
10 /* modify it under the terms of the GNU General Public License */
11 /* as published by the Free Software Foundation. */
12 /************************************************************************/
13 // $Id: gradebook.inc.php 7208 2008-05-28 16:07:24Z cindy $
16 // return: array of grade scales created by member_id, member_id is set to 0 by default, which is to get preset grade scales
17 // returned array format:
22 // [3] => Excellent - Inadequate
23 // [grade_scale_id] => scale_value_max - scale_value_min (Value Explanation)
25 define ('USE_HIGHER_GRADE', 1);
26 define ('USE_LOWER_GRADE', 2);
27 define ('NOT_OVERWRITE', 3);
28 define ('OVERWRITE', 4);
30 function get_grade_scales_array($member_id = 0)
34 $sql = "SELECT d.grade_scale_id, MIN(percentage_to) min, MAX(percentage_to) max FROM ".TABLE_PREFIX."grade_scales_detail d, ".TABLE_PREFIX."grade_scales g WHERE d.grade_scale_id = g.grade_scale_id AND g.member_id = ". $member_id ." GROUP BY d.grade_scale_id";
35 $result = mysql_query($sql, $db) or die(mysql_error());
37 $preset_grade_scales = array();
38 while ($row = mysql_fetch_assoc($result))
40 $sql_min = "SELECT scale_value FROM ".TABLE_PREFIX."grade_scales_detail WHERE grade_scale_id=".$row["grade_scale_id"]." AND percentage_to=".$row["min"];
41 $result_min = mysql_query($sql_min, $db) or die(mysql_error());
42 $row_min = mysql_fetch_assoc($result_min);
43 $min_value = $row_min['scale_value'];
45 $sql_max = "SELECT scale_value FROM ".TABLE_PREFIX."grade_scales_detail WHERE grade_scale_id=".$row["grade_scale_id"]." AND percentage_to=".$row["max"];
46 $result_max = mysql_query($sql_max, $db) or die(mysql_error());
47 $row_max = mysql_fetch_assoc($result_max);
48 $max_value = $row_max['scale_value'];
50 $preset_grade_scales[$row["grade_scale_id"]] = $max_value . " - ". $min_value;
53 return $preset_grade_scales;
56 // generate html of dropdown list box on preset grade scales and grade scales created by current member Id
57 // parameter: $selected_grade_scale_id: the grade_scale_id that need to set to be selected.
59 function print_grade_scale_selectbox($selected_grade_scale_id = 0, $id_name="selected_grade_scale_id")
62 <select name="selected_grade_scale_id" id="<?php echo $id_name; ?>">
63 <option value="0" <?php if ($selected_grade_scale_id == 0) { echo 'selected="selected"'; } ?>><?php echo _AT('none'); ?></option>
65 // preset grade scales
66 $preset_scales = get_grade_scales_array();
68 if (count($preset_scales) > 0)
71 <optgroup label="<?php echo _AT('presets'); ?>">
74 foreach ($preset_scales as $id=>$preset)
76 echo '<option value="'.$id.'" ';
77 if ($selected_grade_scale_id == $id) echo 'selected="selected"';
78 echo '>'.$preset.'</option>'."\n\r";
80 echo ' </optgroup>'."\n\r";
84 $custom_scales = get_grade_scales_array($_SESSION["member_id"]);
86 if (count($custom_scales) > 0)
88 echo ' <optgroup label="'. _AT('custom').'">'."\n\r";
89 foreach ($custom_scales as $id=>$custom)
91 echo '<option value="'.$id.'" ';
92 if ($selected_grade_scale_id == $id) echo 'selected="selected"';
93 echo '>'.$custom.'</option>'."\n\r";
95 echo ' </optgroup>'."\n\r";
103 // This function returns grade based on grade scale
104 // Note: $score can be one of: grade itself, percentage, raw final score.
105 // If $score is raw final score, $out_of has to be provided, otherwise,
106 // don't have to provide $out_of
107 function get_mark_by_grade($grade_scale_id, $score, $out_of='')
111 $score = trim($score);
112 $out_of = trim($out_of);
114 if ($out_of == '') $default_mark = $score;
115 else $default_mark = $score ." / " . $out_of;
117 // if $grade_scale_id is 0 or not given, return $score itself.
118 if ($grade_scale_id == 0 || $grade_scale_id == '')
119 $mark = $default_mark;
122 $sql_grade = "SELECT * from ".TABLE_PREFIX."grade_scales_detail WHERE grade_scale_id = ". $grade_scale_id. " ORDER BY percentage_to DESC";
123 $result_grade = mysql_query($sql_grade, $db) or die(mysql_error());
125 if (mysql_num_rows($result_grade) == 0)
126 $mark = $default_mark;
129 // check if $score is already the grade. If it is, return $score
130 while ($row_grade = mysql_fetch_assoc($result_grade))
132 if ($row_grade['scale_value'] == $score) return $score;
135 if (substr($score, -1) == '%') // percentage
136 $mark_in_percentage = substr($score, 0, -1);
137 else if ($out_of <> '' && $out_of <> 0) // raw final score
138 $mark_in_percentage = $score / $out_of * 100;
140 mysql_data_seek($result_grade, 0);
141 while ($row_grade = mysql_fetch_assoc($result_grade))
143 if ($mark_in_percentage <= $row_grade['percentage_to'] && $mark_in_percentage >= $row_grade['percentage_from'])
144 $mark = $row_grade['scale_value'];
149 // in case grade definition does not cover all scores
150 if ($mark == '') $mark = $default_mark;
155 function get_member_grade($test_id, $member_id, $grade_scale_id)
159 require_once(AT_INCLUDE_PATH.'lib/test_result_functions.inc.php');
163 // find out final_score, out_of
164 $sql = "SELECT t.random, t.out_of, r.result_id, r.final_score FROM ".TABLE_PREFIX."tests t, ".TABLE_PREFIX."tests_results r WHERE t.test_id=".$test_id." AND t.test_id=r.test_id AND r.member_id=".$member_id;
165 $result = mysql_query($sql, $db) or die(mysql_error());
166 $row = mysql_fetch_assoc($result);
168 if (mysql_num_rows($result) > 0 && $row["final_score"] <> "")
170 if ($row['random']) {
171 $out_of = get_random_outof($test_id, $row['result_id']);
173 $out_of = $row['out_of'];
175 $grade = get_mark_by_grade($grade_scale_id, $row["final_score"], $out_of);
181 // Return array of students in the given course who take the given test more than once
182 // Parameter: $member_id, $test_id
183 // Return: an empty array or
186 // [member_id1] => [num_takes1]
187 // [member_id2] => [num_takes2]
190 function get_studs_take_more_than_once($course_id, $test_id)
194 $rtn_array = array();
196 $sql = "SELECT m.first_name, m.last_name, count(result_id) num FROM ".TABLE_PREFIX."members m, ".TABLE_PREFIX."course_enrollment e, ".TABLE_PREFIX."tests_results t WHERE m.member_id = e.member_id AND e.course_id = ".$course_id." AND e.approved='y' AND e.role <> 'Instructor' AND e.member_id=t.member_id AND t.test_id=".$test_id." GROUP BY m.first_name, m.last_name having count(*) > 1";
197 $result = mysql_query($sql, $db) or die(mysql_error());
199 while ($row = mysql_fetch_assoc($result))
200 $rtn_array[$row["first_name"]. " " . $row["last_name"]] = $row["num"];
206 // parameters: 2 grades to compare, grade_scale_id, "higher"/"lower": return higher or lower grade
207 // grade can be percentage like 70% or grade defined in grade_scale_id, like "A", "B"...
208 // return: higher or lower grade depending on 4th parameter
209 // or, -1 if grades are comparable
210 function compare_grades($grade1, $grade2, $gradebook_test_id, $mode = "higher")
214 // get grade scale id
215 $sql = "SELECT grade_scale_id FROM ".TABLE_PREFIX."gradebook_tests WHERE gradebook_test_id = ".$gradebook_test_id;
216 $result = mysql_query($sql, $db) or die(mysql_error());
217 $row = mysql_fetch_assoc($result);
218 $grade_scale_id = $row["grade_scale_id"];
220 if ($grade_scale_id == 0) // compare raw scores
222 // retrieve raw score
223 $grade1 = trim(str_replace("%", "", $grade1));
224 $grade2 = trim(str_replace("%", "", $grade2));
226 if ($grade1 > $grade2) return 1;
227 else if ($grade1 < $grade2) return -1;
232 $grade1 = get_mark_by_grade($grade_scale_id, $grade1);
233 $grade2 = get_mark_by_grade($grade_scale_id, $grade2);
237 $sql_grade = "SELECT scale_value from ".TABLE_PREFIX."grade_scales_detail WHERE grade_scale_id = ". $grade_scale_id. " ORDER BY percentage_to DESC";
238 $result_grade = mysql_query($sql_grade, $db) or die(mysql_error());
239 while ($row_grade = mysql_fetch_assoc($result_grade))
241 $grades[] = $row_grade["scale_value"];
244 if (!in_array($grade1, $grades) || !in_array($grade2, $grades))
245 return -1; // uncomparable
248 $grade1_key = array_search($grade1, $grades);
249 $grade2_key = array_search($grade2, $grades);
251 if ($grade1_key > $grade2_key)
253 $higher_grade = $grade2;
254 $lower_grade = $grade1;
256 else if ($grade1_key < $grade2_key)
258 $higher_grade = $grade1;
259 $lower_grade = $grade2;
261 else $higher_grade = $lower_grade = $grade1;
265 if ($mode == "higher") return $higher_grade;
266 else return $lower_grade;
269 // check imported students and grades:
270 // 1. if the student exists in the class, if not, report error
271 // 2. if the grade already exists, if it is, report conflict
272 // parameter: an array of student/grade info
276 // [fname] => angelo (could be empty if [member_id] is given)
277 // [lname] => yuan (could be empty if [member_id] is given)
278 // [email] => angelo@hotmail.com (could be empty if [member_id] is given)
280 // [gradebook_test_id] => 4
281 // [solve_conflict] => 0
283 // return: an array of processed student/grade/error info
287 // [fname] => angelo (could be empty if [member_id] is given)
288 // [lname] => yuan (could be empty if [member_id] is given)
289 // [email] => angelo@hotmail.com (could be empty if [member_id] is given)
291 // [gradebook_test_id] => 4
292 // [solve_conflict] => 0
293 // [error] => "Student not exists"
294 // [has_conflict] => 1
296 function check_user_info($record)
300 $record['fname'] = htmlspecialchars(stripslashes(trim($record['fname'])));
301 $record['lname'] = htmlspecialchars(stripslashes(trim($record['lname'])));
302 $record['member_id'] = htmlspecialchars(stripslashes(trim($record['member_id'])));
303 $record['email'] = htmlspecialchars(stripslashes(trim($record['email'])));
304 $record['grade'] = htmlspecialchars(stripslashes(trim($record['grade'])));
306 if (empty($record['remove'])) {
307 $record['remove'] = FALSE;
310 if ($record['member_id'] == '')
312 $sql = "SELECT * FROM ".TABLE_PREFIX."members m, ".TABLE_PREFIX."course_enrollment e WHERE m.first_name='".$record['fname']."' AND m.last_name='".$record['lname']."' AND m.email='".$record['email']."' AND m.member_id = e.member_id AND e.course_id=".$_SESSION["course_id"]." AND e.approved='y' AND e.role<>'Instructor'";
313 $result = mysql_query($sql, $db) or die(mysql_error());
315 if (mysql_num_rows($result) == 0)
316 $record['error'] = _AT("student_not_exists");
319 $row = mysql_fetch_assoc($result);
320 $record['member_id'] = $row["member_id"];
324 if ($record['error'] == "" && $record['member_id'] > 0)
326 $sql = "SELECT grade FROM ".TABLE_PREFIX."gradebook_detail WHERE gradebook_test_id=".$record['gradebook_test_id']. " AND member_id=".$record["member_id"];
327 $result = mysql_query($sql, $db) or die(mysql_error());
329 if (mysql_num_rows($result) > 0 && $record['solve_conflict'] == 0)
331 $row = mysql_fetch_assoc($result);
332 $record['error'] = _AT("grade_already_exists", $row["grade"]);
333 $record['conflict'] = 1;
336 if (mysql_num_rows($result) > 0 && $record['solve_conflict'] > 0)
338 $row = mysql_fetch_assoc($result);
340 if ($record['solve_conflict'] == USE_HIGHER_GRADE || $record['solve_conflict'] == USE_LOWER_GRADE)
342 if ($record['solve_conflict'] == USE_HIGHER_GRADE)
343 $grade = compare_grades($record['grade'], $row['grade'], $record['gradebook_test_id'], "higher");
345 if ($record['solve_conflict'] == USE_LOWER_GRADE)
346 $grade = compare_grades($record['grade'], $row['grade'], $record['gradebook_test_id'], "lower");
350 $record["error"] = _AT("grades_uncomparable");
351 $record['conflict'] = 1;
354 $record['grade'] = $grade;
357 if ($record['solve_conflict'] == NOT_OVERWRITE) $record['grade'] = $row['grade'];
358 if ($record['solve_conflict'] == OVERWRITE) $record['grade'] = $record['grade'];
362 if ($record['remove']) {
364 $record['error'] = '';
371 function update_gradebook_external_test($students, $gradebook_test_id)
375 foreach($students as $student)
377 if (!$student['remove'])
379 // retrieve member id
380 $sql = "SELECT member_id FROM ".TABLE_PREFIX."members m WHERE m.first_name='".$student['fname']."' AND m.last_name='".$student['lname']."' AND m.email='".$student['email']."'";
381 $result = mysql_query($sql, $db) or die(mysql_error());
382 $row = mysql_fetch_assoc($result);
383 $member_id = $row["member_id"];
385 // retrieve grade scale id
386 $sql = "SELECT grade_scale_id FROM ".TABLE_PREFIX."gradebook_tests WHERE gradebook_test_id=".$gradebook_test_id;
387 $result = mysql_query($sql, $db) or die(mysql_error());
388 $row = mysql_fetch_assoc($result);
389 $grade_scale_id = $row["grade_scale_id"];
391 $grade = get_mark_by_grade($grade_scale_id, $student["grade"]);
392 $sql = "REPLACE INTO ".TABLE_PREFIX."gradebook_detail(gradebook_test_id, member_id, grade) VALUES(".$gradebook_test_id.", ".$member_id.", '".$grade."')";
393 $result = mysql_query($sql, $db) or die(mysql_error());
395 $updated_list .= '<li>' . $student['fname'] . ', '. $student['lname']. ': '. $grade. '</li>';
401 $feedback = array('GRADEBOOK_UPDATED', $updated_list);
402 $msg->addFeedback($feedback);
406 // return median value of the given array
407 function median($grade_array)
409 $oe_value = count($grade_array);
411 if ($oe_value % 2 == 0 )
417 $median = $grade_array[(count($grade_array)/2)];
419 $median= $grade_array[(count($grade_array)/2)-1];
424 // return class average of the given test_id
425 function get_class_avg($gradebook_test_id)
429 $sql = "SELECT * FROM ".TABLE_PREFIX."gradebook_tests WHERE gradebook_test_id=".$gradebook_test_id;
430 $result = mysql_query($sql, $db) or die(mysql_error());
431 $row = mysql_fetch_assoc($result);
433 if ($row["id"]<>0) // internal atutor test
435 require_once (AT_INCLUDE_PATH.'lib/test_result_functions.inc.php');
437 $sql_test = "SELECT * FROM ".TABLE_PREFIX."tests WHERE test_id=".$row["id"];
438 $result_test = mysql_query($sql_test, $db) or die(mysql_error());
439 $row_test = mysql_fetch_assoc($result_test);
441 if ($row_test['out_of'] == 0 || $row_test['result_release']==AT_RELEASE_NEVER)
444 $sql_marks = "SELECT * FROM ".TABLE_PREFIX."tests_results WHERE test_id=".$row["id"]. " AND status=1";
445 $result_marks = mysql_query($sql_marks, $db) or die(mysql_error());
448 $total_final_score = 0;
450 while ($row_marks = mysql_fetch_assoc($result_marks))
452 if ($row_marks['final_score'] == '' ) continue;
455 $total_final_score += $row_marks["final_score"];
457 if ($row_test['random'])
458 $total_out_of += get_random_outof($row_marks['test_id'], $row_marks['result_id']);
460 $total_out_of += $row_test['out_of'];
463 if ($num_students > 0)
465 $avg_final_score = round($total_final_score / $num_students);
466 $avg_out_of = round($total_out_of / $num_students);
469 if ($avg_final_score <> "")
470 $avg_grade = get_mark_by_grade($row["grade_scale_id"], $avg_final_score, $avg_out_of);
474 else // external test
476 $sql_grades = "SELECT * FROM ".TABLE_PREFIX."gradebook_detail WHERE gradebook_test_id=".$gradebook_test_id." ORDER BY grade";
477 $result_grades = mysql_query($sql_grades, $db) or die(mysql_error());
479 $grade_array = array();
480 while ($row_grades = mysql_fetch_assoc($result_grades))
481 $grade_array[] = $row_grades["grade"];
483 $avg_grade = median($grade_array);
486 if ($avg_grade == "") return _AT("na");
487 else return $avg_grade;