123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625 |
- <?php
- /* For licensing terms, see /license.txt */
- /**
- * Class ScoreDisplay
- * Display scores according to the settings made by the platform admin.
- * This class works as a singleton: call instance() to retrieve an object.
- *
- * @author Bert Steppé
- *
- * @package chamilo.gradebook
- */
- class ScoreDisplay
- {
- private $coloring_enabled;
- private $color_split_value;
- private $custom_enabled;
- private $upperlimit_included;
- private $custom_display;
- private $custom_display_conv;
- /**
- * Protected constructor - call instance() to instantiate.
- *
- * @param int $category_id
- */
- public function __construct($category_id = 0)
- {
- if (!empty($category_id)) {
- $this->category_id = $category_id;
- }
- // Loading portal settings + using standard functions.
- $value = api_get_setting('gradebook_score_display_coloring');
- // Setting coloring.
- $this->coloring_enabled = $value == 'true' ? true : false;
- if ($this->coloring_enabled) {
- $value = api_get_setting('gradebook_score_display_colorsplit');
- if (isset($value)) {
- $this->color_split_value = $value;
- }
- }
- // Setting custom enabled
- $value = api_get_setting('gradebook_score_display_custom');
- $this->custom_enabled = $value == 'true' ? true : false;
- if ($this->custom_enabled) {
- $params = ['category = ?' => ['Gradebook']];
- $displays = api_get_settings_params($params);
- $portal_displays = [];
- if (!empty($displays)) {
- foreach ($displays as $display) {
- $data = explode('::', $display['selected_value']);
- if (empty($data[1])) {
- $data[1] = '';
- }
- $portal_displays[$data[0]] = [
- 'score' => $data[0],
- 'display' => $data[1],
- ];
- }
- sort($portal_displays);
- }
- $this->custom_display = $portal_displays;
- if (count($this->custom_display) > 0) {
- $value = api_get_setting('gradebook_score_display_upperlimit');
- $value = $value['my_display_upperlimit'];
- $this->upperlimit_included = $value == 'true' ? true : false;
- $this->custom_display_conv = $this->convert_displays($this->custom_display);
- }
- }
- //If teachers can override the portal parameters
- if (api_get_setting('teachers_can_change_score_settings') == 'true') {
- //Load course settings
- if ($this->custom_enabled) {
- $this->custom_display = $this->get_custom_displays();
- if (count($this->custom_display) > 0) {
- $this->custom_display_conv = $this->convert_displays($this->custom_display);
- }
- }
- if ($this->coloring_enabled) {
- $this->color_split_value = $this->get_score_color_percent();
- }
- }
- }
- /**
- * Get the instance of this class.
- *
- * @param int $category_id
- *
- * @return ScoreDisplay
- */
- public static function instance($category_id = 0)
- {
- static $instance;
- if (!isset($instance)) {
- $instance = new ScoreDisplay($category_id);
- }
- return $instance;
- }
- /**
- * Compare the custom display of 2 scores, can be useful in sorting.
- */
- public static function compare_scores_by_custom_display($score1, $score2)
- {
- if (!isset($score1)) {
- return isset($score2) ? 1 : 0;
- } elseif (!isset($score2)) {
- return -1;
- } else {
- $scoreDisplay = self::instance();
- $custom1 = $scoreDisplay->display_custom($score1);
- $custom2 = $scoreDisplay->display_custom($score2);
- if ($custom1 == $custom2) {
- return 0;
- } else {
- return ($score1[0] / $score1[1]) < ($score2[0] / $score2[1]) ? -1 : 1;
- }
- }
- }
- /**
- * Is coloring enabled ?
- */
- public function is_coloring_enabled()
- {
- return $this->coloring_enabled;
- }
- /**
- * Is custom score display enabled ?
- */
- public function is_custom()
- {
- return $this->custom_enabled;
- }
- /**
- * Is upperlimit included ?
- */
- public function is_upperlimit_included()
- {
- return $this->upperlimit_included;
- }
- /**
- * If custom score display is enabled, this will return the current settings.
- * See also update_custom_score_display_settings.
- *
- * @return array current settings (or null if feature not enabled)
- */
- public function get_custom_score_display_settings()
- {
- return $this->custom_display;
- }
- /**
- * If coloring is enabled, scores below this value will be displayed in red.
- *
- * @return int color split value, in percent (or null if feature not enabled)
- */
- public function get_color_split_value()
- {
- return $this->color_split_value;
- }
- /**
- * Update custom score display settings.
- *
- * @param array $displays 2-dimensional array - every sub array must have keys (score, display)
- * @param int score color percent (optional)
- * @param int gradebook category id (optional)
- */
- public function update_custom_score_display_settings(
- $displays,
- $scorecolpercent = 0,
- $category_id = null
- ) {
- $this->custom_display = $displays;
- $this->custom_display_conv = $this->convert_displays($this->custom_display);
- if (isset($category_id)) {
- $category_id = intval($category_id);
- } else {
- $category_id = $this->get_current_gradebook_category_id();
- }
- // remove previous settings
- $table = Database::get_main_table(TABLE_MAIN_GRADEBOOK_SCORE_DISPLAY);
- $sql = 'DELETE FROM '.$table.' WHERE category_id = '.$category_id;
- Database::query($sql);
- // add new settings
- $count = 0;
- foreach ($displays as $display) {
- $params = [
- 'score' => $display['score'],
- 'display' => $display['display'],
- 'category_id' => $category_id,
- 'score_color_percent' => $scorecolpercent,
- ];
- Database::insert($table, $params);
- $count++;
- }
- }
- /**
- * @param int $category_id
- *
- * @return false|null
- */
- public function insert_defaults($category_id)
- {
- if (empty($category_id)) {
- return false;
- }
- //Get this from DB settings
- $display = [
- 50 => get_lang('Failed'),
- 60 => get_lang('Poor'),
- 70 => get_lang('Fair'),
- 80 => get_lang('Good'),
- 90 => get_lang('Outstanding'),
- 100 => get_lang('Excellent'),
- ];
- $table = Database::get_main_table(TABLE_MAIN_GRADEBOOK_SCORE_DISPLAY);
- foreach ($display as $value => $text) {
- $params = [
- 'score' => $value,
- 'display' => $text,
- 'category_id' => $category_id,
- 'score_color_percent' => 0,
- ];
- Database::insert($table, $params);
- }
- }
- /**
- * @return int
- */
- public function get_number_decimals()
- {
- $number_decimals = api_get_setting('gradebook_number_decimals');
- if (!isset($number_decimals)) {
- $number_decimals = 0;
- }
- return $number_decimals;
- }
- /**
- * Formats a number depending of the number of decimals.
- *
- * @param float $score
- * @param bool $ignoreDecimals
- * @param string $decimalSeparator
- * @param string $thousandSeparator
- *
- * @return float the score formatted
- */
- public function format_score($score, $ignoreDecimals = false, $decimalSeparator = '.', $thousandSeparator = ',')
- {
- $decimals = $this->get_number_decimals();
- if ($ignoreDecimals) {
- $decimals = 0;
- }
- return api_number_format($score, $decimals, $decimalSeparator, $thousandSeparator);
- }
- /**
- * Display a score according to the current settings.
- *
- * @param array $score data structure, as returned by the calc_score functions
- * @param int $type one of the following constants:
- * SCORE_DIV, SCORE_PERCENT, SCORE_DIV_PERCENT, SCORE_AVERAGE
- * (ignored for student's view if custom score display is enabled)
- * @param int $what one of the following constants:
- * SCORE_BOTH, SCORE_ONLY_DEFAULT, SCORE_ONLY_CUSTOM (default: SCORE_BOTH)
- * (only taken into account if custom score display is enabled and for course/platform admin)
- * @param bool $disableColor
- * @param bool $ignoreDecimals
- *
- * @return string
- */
- public function display_score(
- $score,
- $type = SCORE_DIV_PERCENT,
- $what = SCORE_BOTH,
- $disableColor = false,
- $ignoreDecimals = false
- ) {
- $my_score = $score == 0 ? 1 : $score;
- if ($type == SCORE_BAR) {
- $percentage = $my_score[0] / $my_score[1] * 100;
- return Display::bar_progress($percentage);
- }
- if ($type == SCORE_NUMERIC) {
- $percentage = $my_score[0] / $my_score[1] * 100;
- return round($percentage);
- }
- if ($type == SCORE_SIMPLE) {
- $simpleScore = $this->format_score($my_score[0], $ignoreDecimals);
- return $simpleScore;
- }
- if ($this->custom_enabled && isset($this->custom_display_conv)) {
- $display = $this->display_default($my_score, $type, $ignoreDecimals);
- } else {
- // if no custom display set, use default display
- $display = $this->display_default($my_score, $type, $ignoreDecimals);
- }
- if ($this->coloring_enabled && $disableColor == false) {
- $my_score_denom = isset($score[1]) && !empty($score[1]) && $score[1] > 0 ? $score[1] : 1;
- $scoreCleaned = isset($score[0]) ? $score[0] : 0;
- if (($scoreCleaned / $my_score_denom) < ($this->color_split_value / 100)) {
- $display = Display::tag(
- 'font',
- $display,
- ['color' => 'red']
- );
- }
- }
- return $display;
- }
- /**
- * Get current gradebook category id.
- *
- * @return int Category id
- */
- private function get_current_gradebook_category_id()
- {
- $table = Database::get_main_table(TABLE_MAIN_GRADEBOOK_CATEGORY);
- $courseId = api_get_course_int_id();
- $curr_session_id = api_get_session_id();
- if (empty($curr_session_id)) {
- $session_condition = ' AND session_id is null ';
- } else {
- $session_condition = ' AND session_id = '.$curr_session_id;
- }
- $sql = 'SELECT id FROM '.$table.'
- WHERE c_id = "'.$courseId.'" '.$session_condition;
- $rs = Database::query($sql);
- $category_id = 0;
- if (Database::num_rows($rs) > 0) {
- $row = Database::fetch_row($rs);
- $category_id = $row[0];
- }
- return $category_id;
- }
- /**
- * @param $score
- * @param int $type
- * @param bool $ignoreDecimals
- *
- * @return string
- */
- private function display_default($score, $type, $ignoreDecimals = false)
- {
- switch ($type) {
- case SCORE_DIV: // X / Y
- return $this->display_as_div($score, $ignoreDecimals);
- case SCORE_PERCENT: // XX %
- return $this->display_as_percent($score);
- case SCORE_DIV_PERCENT: // X / Y (XX %)
- return $this->display_as_div($score).' ('.$this->display_as_percent($score).')';
- case SCORE_AVERAGE: // XX %
- return $this->display_as_percent($score);
- case SCORE_DECIMAL: // 0.50 (X/Y)
- return $this->display_as_decimal($score);
- case SCORE_DIV_PERCENT_WITH_CUSTOM: // X / Y (XX %) - Good!
- $custom = $this->display_custom($score);
- if (!empty($custom)) {
- $custom = ' - '.$custom;
- }
- return $this->display_as_div($score).' ('.$this->display_as_percent($score).')'.$custom;
- case SCORE_DIV_SIMPLE_WITH_CUSTOM: // X - Good!
- $custom = $this->display_custom($score);
- if (!empty($custom)) {
- $custom = ' - '.$custom;
- }
- return $this->display_simple_score($score).$custom;
- break;
- case SCORE_DIV_SIMPLE_WITH_CUSTOM_LETTERS:
- $custom = $this->display_custom($score);
- if (!empty($custom)) {
- $custom = ' - '.$custom;
- }
- $score = $this->display_simple_score($score);
- //needs sudo apt-get install php5-intl
- if (class_exists('NumberFormatter')) {
- $iso = api_get_language_isocode();
- $f = new NumberFormatter($iso, NumberFormatter::SPELLOUT);
- $letters = $f->format($score);
- $letters = api_strtoupper($letters);
- $letters = " ($letters) ";
- }
- return $score.$letters.$custom;
- break;
- case SCORE_CUSTOM: // Good!
- return $this->display_custom($score);
- }
- }
- /**
- * @param array $score
- *
- * @return float|string
- */
- private function display_simple_score($score)
- {
- if (isset($score[0])) {
- return $this->format_score($score[0]);
- }
- return '';
- }
- /**
- * Returns "1" for array("100", "100");.
- *
- * @param array $score
- *
- * @return float
- */
- private function display_as_decimal($score)
- {
- $score_denom = ($score[1] == 0) ? 1 : $score[1];
- return $this->format_score($score[0] / $score_denom);
- }
- /**
- * Returns "100 %" for array("100", "100");.
- */
- private function display_as_percent($score)
- {
- $score_denom = ($score[1] == 0) ? 1 : $score[1];
- return $this->format_score($score[0] / $score_denom * 100).' %';
- }
- /**
- * Returns 10.00 / 10.00 for array("100", "100");.
- *
- * @param array $score
- * @param bool $ignoreDecimals
- *
- * @return string
- */
- private function display_as_div($score, $ignoreDecimals = false)
- {
- if ($score == 1) {
- return '0 / 0';
- } else {
- $score[0] = isset($score[0]) ? $this->format_score($score[0], $ignoreDecimals) : 0;
- $score[1] = isset($score[1]) ? $this->format_score($score[1], $ignoreDecimals) : 0;
- return $score[0].' / '.$score[1];
- }
- }
- /**
- * Depends on the teacher's configuration of thresholds. i.e. [0 50] "Bad", [50:100] "Good".
- *
- * @param array $score
- *
- * @return string
- */
- private function display_custom($score)
- {
- $my_score_denom = $score[1] == 0 ? 1 : $score[1];
- $scaledscore = $score[0] / $my_score_denom;
- if ($this->upperlimit_included) {
- foreach ($this->custom_display_conv as $displayitem) {
- if ($scaledscore <= $displayitem['score']) {
- return $displayitem['display'];
- }
- }
- } else {
- if (!empty($this->custom_display_conv)) {
- foreach ($this->custom_display_conv as $displayitem) {
- if ($scaledscore < $displayitem['score'] || $displayitem['score'] == 1) {
- return $displayitem['display'];
- }
- }
- }
- }
- }
- /**
- * Get score color percent by category.
- *
- * @param int Gradebook category id
- *
- * @return int Score
- */
- private function get_score_color_percent($category_id = null)
- {
- $tbl_display = Database::get_main_table(TABLE_MAIN_GRADEBOOK_SCORE_DISPLAY);
- if (isset($category_id)) {
- $category_id = intval($category_id);
- } else {
- $category_id = $this->get_current_gradebook_category_id();
- }
- $sql = 'SELECT score_color_percent FROM '.$tbl_display.'
- WHERE category_id = '.$category_id.'
- LIMIT 1';
- $result = Database::query($sql);
- $score = 0;
- if (Database::num_rows($result) > 0) {
- $row = Database::fetch_row($result);
- $score = $row[0];
- }
- return $score;
- }
- /**
- * Get current custom score display settings.
- *
- * @param int Gradebook category id
- *
- * @return array 2-dimensional array every element contains 3 subelements (id, score, display)
- */
- private function get_custom_displays($category_id = null)
- {
- $tbl_display = Database::get_main_table(TABLE_MAIN_GRADEBOOK_SCORE_DISPLAY);
- if (isset($category_id)) {
- $category_id = intval($category_id);
- } else {
- $category_id = $this->get_current_gradebook_category_id();
- }
- $sql = 'SELECT * FROM '.$tbl_display.'
- WHERE category_id = '.$category_id.'
- ORDER BY score';
- $result = Database::query($sql);
- return Database::store_result($result, 'ASSOC');
- }
- /**
- * Convert display settings to internally used values.
- */
- private function convert_displays($custom_display)
- {
- if (isset($custom_display)) {
- // get highest score entry, and copy each element to a new array
- $converted = [];
- $highest = 0;
- foreach ($custom_display as $element) {
- if ($element['score'] > $highest) {
- $highest = $element['score'];
- }
- $converted[] = $element;
- }
- // sort the new array (ascending)
- usort($converted, ['ScoreDisplay', 'sort_display']);
- // adjust each score in such a way that
- // each score is scaled between 0 and 1
- // the highest score in this array will be equal to 1
- $converted2 = [];
- foreach ($converted as $element) {
- $newelement = [];
- if (isset($highest) && !empty($highest) && $highest > 0) {
- $newelement['score'] = $element['score'] / $highest;
- } else {
- $newelement['score'] = 0;
- }
- $newelement['display'] = $element['display'];
- $converted2[] = $newelement;
- }
- return $converted2;
- } else {
- return null;
- }
- }
- /**
- * @param array $item1
- * @param array $item2
- *
- * @return int
- */
- private function sort_display($item1, $item2)
- {
- if ($item1['score'] === $item2['score']) {
- return 0;
- } else {
- return $item1['score'] < $item2['score'] ? -1 : 1;
- }
- }
- }
|