Chamilo is a learning management system focused on ease of use and accessibility
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 
chamilo-lms/main/inc/lib/extra_field_option.lib.php

927 lines
30 KiB

<?php
/* For licensing terms, see /license.txt */
use Chamilo\CoreBundle\Entity\ExtraFieldOptions;
/**
* Class ExtraFieldOption
* Handles the extra fields for various objects (users, sessions, courses).
*/
class ExtraFieldOption extends Model
{
public $columns = [
'id',
'field_id',
'option_value',
'display_text',
'option_order',
'priority',
'priority_message',
'tms',
];
public $extraField;
public $fieldId;
/**
* Gets the table for the type of object for which we are using an extra field.
*
* @param string $type Type of object (course, user or session)
*/
public function __construct($type)
{
parent::__construct();
$this->type = $type;
$extraField = new ExtraField($this->type);
$this->extraField = $extraField;
$this->table = Database::get_main_table(TABLE_EXTRA_FIELD_OPTIONS);
$this->tableExtraField = Database::get_main_table(TABLE_EXTRA_FIELD);
}
/**
* @return ExtraField
*/
public function getExtraField()
{
return $this->extraField;
}
/**
* Gets the number of options already available in the table for this item type.
*
* @return int Number of options available
* @assert () >= 0
*/
/*public function get_count()
{
$row = Database::select('count(*) as count', $this->table, array(), 'first');
return $row['count'];
}*/
/**
* Gets the number of options available for this field.
*
* @param int $fieldId
*
* @return int Number of options
* @assert ('') === false
* @assert (-1) == 0
* @assert (0) == 0
*/
public function get_count_by_field_id($fieldId)
{
if (empty($fieldId)) {
return false;
}
$extraFieldType = $this->getExtraField()->getExtraFieldType();
$fieldId = intval($fieldId);
$sql = "SELECT count(*) as count
FROM $this->table o
INNER JOIN $this->tableExtraField e
ON o.field_id = e.id
WHERE
o.field_id = $fieldId AND
e.extra_field_type = $extraFieldType ";
$result = Database::query($sql);
$result = Database::fetch_array($result);
return $result['count'];
}
/**
* Returns a list of options for a specific field, separated by ";".
*
* @param int $field_id
* @param bool $add_id_in_array Indicates whether we want the results to be given with their id
* @param string $ordered_by Order by clause (without the "order by") to be added to the SQL query
*
* @return string List of options separated by ;
* @assert (-1, false, null) == ''
*/
public function getFieldOptionsToString($field_id, $add_id_in_array = false, $ordered_by = null)
{
$options = self::get_field_options_by_field($field_id, $add_id_in_array, $ordered_by);
$new_options = [];
if (!empty($options)) {
foreach ($options as $option) {
$new_options[] = $option['option_value'].':'.$option['display_text'];
}
$string = implode(';', $new_options);
return $string;
}
return '';
}
/**
* Delete all the options of a specific field.
*
* @param int $field_id
*
* @assert (-1) === false
*/
public function delete_all_options_by_field_id($field_id)
{
$field_id = intval($field_id);
$sql = "DELETE FROM {$this->table} WHERE field_id = $field_id";
$r = Database::query($sql);
return $r;
}
/**
* @param array $params
* @param bool $showQuery
*
* @return int|bool
*/
public function saveOptions($params, $showQuery = false)
{
$optionInfo = $this->get_field_option_by_field_and_option(
$params['field_id'],
$params['option_value']
);
if ($optionInfo == false) {
$optionValue = api_replace_dangerous_char($params['option_value']);
$order = $this->get_max_order($params['field_id']);
$newParams = [
'field_id' => $params['field_id'],
'value' => trim($optionValue),
'display_text' => trim($params['display_text']),
'option_order' => $order,
];
return parent::save($newParams, $showQuery);
}
return false;
}
/**
* Saves an option into the corresponding *_field_options table.
*
* @param array $params Parameters to be considered for the insertion
* @param bool $showQuery Whether to show the query (sent to the parent save() method)
*
* @return bool True on success, false on error
* @assert (array('field_id'=>0), false) === false
* @assert (array('field_id'=>1), false) === true
*/
public function save($params, $showQuery = false)
{
$field_id = intval($params['field_id']);
if (empty($field_id)) {
return false;
}
$parseOptions = in_array(
$params['field_type'],
[
ExtraField::FIELD_TYPE_RADIO,
ExtraField::FIELD_TYPE_SELECT,
ExtraField::FIELD_TYPE_SELECT_MULTIPLE,
ExtraField::FIELD_TYPE_DOUBLE_SELECT,
ExtraField::FIELD_TYPE_SELECT_WITH_TEXT_FIELD,
ExtraField::FIELD_TYPE_TRIPLE_SELECT,
]
);
if (empty($params['field_options']) || !$parseOptions) {
return true;
}
switch ($params['field_type']) {
case ExtraField::FIELD_TYPE_DOUBLE_SELECT:
//$params['field_options'] = France:Paris;Bretagne;Marseilles;Lyon|Belgique:Bruxelles;Namur;Liège;Bruges|Peru:Lima;Piura;
case ExtraField::FIELD_TYPE_SELECT_WITH_TEXT_FIELD:
//$params['field_options'] = Option 1|Option 2|Option 3
$options_parsed = ExtraField::extra_field_double_select_convert_string_to_array(
$params['field_options']
);
if (empty($options_parsed)) {
break;
}
foreach ($options_parsed as $key => $option) {
$new_params = [
'field_id' => $field_id,
'option_value' => 0,
'display_text' => $option['label'],
'option_order' => 0,
];
// Looking if option already exists:
$option_info = self::get_field_option_by_field_id_and_option_display_text(
$field_id,
$option['label']
);
if (empty($option_info)) {
$sub_id = parent::save($new_params, $showQuery);
} else {
$sub_id = $option_info['id'];
$new_params['id'] = $sub_id;
parent::update($new_params, $showQuery);
}
if ($params['field_type'] == ExtraField::FIELD_TYPE_SELECT_WITH_TEXT_FIELD) {
continue;
}
foreach ($option['options'] as $sub_option) {
if (empty($sub_option)) {
continue;
}
$new_params = [
'field_id' => $field_id,
'option_value' => $sub_id,
'display_text' => $sub_option,
'option_order' => 0,
];
$option_info = self::getFieldOptionByFieldIdAndOptionDisplayTextAndOptionValue(
$field_id,
$sub_option,
$sub_id
);
if (empty($option_info)) {
parent::save($new_params, $showQuery);
continue;
}
$new_params['id'] = $option_info['id'];
parent::update($new_params, $showQuery);
}
}
break;
case ExtraField::FIELD_TYPE_TRIPLE_SELECT:
//Format: Option1\Option11:Option111;Option112\Option12:Option121|Option2\Option21:Option211
$options = ExtraField::tripleSelectConvertStringToArray($params['field_options']);
if (!$options) {
break;
}
foreach ($options as $level1) {
$level1Params = [
'field_id' => $field_id,
'option_value' => 0,
'display_text' => $level1['label'],
'option_order' => 0,
];
$optionInfo = self::get_field_option_by_field_id_and_option_display_text(
$field_id,
$level1['label']
);
if (empty($optionInfo)) {
$level1Id = parent::save($level1Params);
} else {
$level1Id = $optionInfo['id'];
$level1Params['id'] = $level1Id;
parent::update($level1Params);
}
foreach ($level1['options'] as $level2) {
$level2Params = [
'field_id' => $field_id,
'option_value' => $level1Id,
'display_text' => $level2['label'],
'display_order' => 0,
];
$optionInfo = self::getFieldOptionByFieldIdAndOptionDisplayTextAndOptionValue(
$field_id,
$level2['label'],
$level1Id
);
if (empty($optionInfo)) {
$level2Id = parent::save($level2Params);
} else {
$level2Id = $optionInfo['id'];
$level2Params['id'] = $level2Id;
parent::update($level2Params);
}
foreach ($level2['options'] as $level3) {
foreach ($level3 as $item) {
$level3Params = [
'field_id' => $field_id,
'option_value' => $level2Id,
'display_text' => $item,
'display_order' => 0,
];
$optionInfo = self::getFieldOptionByFieldIdAndOptionDisplayTextAndOptionValue(
$field_id,
$item,
$level2Id
);
if (empty($optionInfo)) {
parent::save($level3Params);
} else {
$level3Params['id'] = $optionInfo['id'];
parent::update($level3Params);
}
}
}
}
}
break;
default:
$list = explode(';', $params['field_options']);
foreach ($list as $option) {
$option_info = $this->get_field_option_by_field_and_option($field_id, $option);
// Use URLify only for new items
$optionValue = api_replace_dangerous_char($option);
$option = trim($option);
if ($option_info != false) {
continue;
}
$order = $this->get_max_order($field_id);
$new_params = [
'field_id' => $field_id,
'option_value' => trim($optionValue),
'display_text' => trim($option),
'option_order' => $order,
];
parent::save($new_params, $showQuery);
}
break;
}
return true;
}
/**
* Save one option item at a time.
*
* @param array $params Parameters specific to the option
* @param bool $show_query Whether to show the query (sent to parent save() method)
* @param bool $insert_repeated Whether to insert even if the option already exists
*
* @return bool True on success, false on failure
* @assert (array('field_id'=>0),false) === false
* @assert (array('field_id'=>0),false) === true
*/
public function save_one_item($params, $show_query = false, $insert_repeated = true)
{
$field_id = intval($params['field_id']);
if (empty($field_id)) {
return false;
}
if (isset($params['option_value'])) {
$params['option_value'] = trim($params['option_value']);
}
if (isset($params['display_text'])) {
$params['display_text'] = trim($params['display_text']);
}
if (empty($params['option_order'])) {
$order = $this->get_max_order($field_id);
$params['option_order'] = $order;
}
if ($insert_repeated) {
parent::save($params, $show_query);
} else {
$check = $this->get_field_option_by_field_and_option(
$field_id,
$params['option_value']
);
if ($check == false) {
parent::save($params, $show_query);
}
}
return true;
}
/**
* Get the complete row of a specific option of a specific field.
*
* @param int $field_id
* @param string $option_value Value of the option
*
* @return mixed The row on success or false on failure
* @assert (0,'') === false
*/
public function get_field_option_by_field_and_option($field_id, $option_value)
{
$field_id = intval($field_id);
$option_value = Database::escape_string($option_value);
$extraFieldType = $this->getExtraField()->getExtraFieldType();
$sql = "SELECT s.* FROM {$this->table} s
INNER JOIN {$this->tableExtraField} sf
ON (s.field_id = sf.id)
WHERE
field_id = $field_id AND
option_value = '".$option_value."' AND
sf.extra_field_type = $extraFieldType
";
$result = Database::query($sql);
if (Database::num_rows($result) > 0) {
return Database::store_result($result, 'ASSOC');
}
return false;
}
/**
* Get the complete row of a specific option's display text of a specific field.
*
* @param int $field_id
* @param string $option_display_text Display value of the option
*
* @return mixed The row on success or false on failure
* @assert (0, '') === false
*/
public function get_field_option_by_field_id_and_option_display_text($field_id, $option_display_text)
{
$field_id = intval($field_id);
$option_display_text = Database::escape_string($option_display_text);
$extraFieldType = $this->getExtraField()->getExtraFieldType();
$sql = "SELECT s.* FROM {$this->table} s
INNER JOIN {$this->tableExtraField} sf
ON (s.field_id = sf.id)
WHERE
field_id = $field_id AND
s.display_text = '".$option_display_text."' AND
sf.extra_field_type = $extraFieldType
";
$result = Database::query($sql);
if (Database::num_rows($result) > 0) {
return Database::fetch_array($result, 'ASSOC');
}
return false;
}
/**
* Get the complete row of a specific option's display text of a specific field.
*
* @param int $field_id
* @param string $option_display_text Display value of the option
* @param string $option_value Value of the option
*
* @return mixed The row on success or false on failure
* @assert (0, '', '') === false
*/
public function getFieldOptionByFieldIdAndOptionDisplayTextAndOptionValue(
$field_id,
$option_display_text,
$option_value
) {
$field_id = intval($field_id);
$option_display_text = Database::escape_string($option_display_text);
$option_value = Database::escape_string($option_value);
$extraFieldType = $this->getExtraField()->getExtraFieldType();
$sql = "SELECT s.* FROM {$this->table} s
INNER JOIN {$this->tableExtraField} sf
ON (s.field_id = sf.id)
WHERE
field_id = $field_id AND
sf.display_text = '".$option_display_text."' AND
option_value = '$option_value' AND
sf.extra_field_type = ".$extraFieldType."
";
$result = Database::query($sql);
if (Database::num_rows($result) > 0) {
return Database::fetch_array($result, 'ASSOC');
}
return false;
}
/**
* Gets an array of options for a specific field.
*
* @param int $field_id The field ID
* @param bool $add_id_in_array Whether to add the row ID in the result
* @param null $ordered_by Extra ordering query bit
*
* @return array The options if they exists. Otherwise return false
*/
public function get_field_options_by_field($field_id, $add_id_in_array = false, $ordered_by = null)
{
$field_id = intval($field_id);
$orderBy = null;
switch ($ordered_by) {
case 'id':
$orderBy = ['id' => 'ASC'];
break;
case 'field_id':
$orderBy = ['fieldId' => 'ASC'];
break;
case 'option_value':
$orderBy = ['optionValue' => 'ASC'];
break;
case 'display_text':
$orderBy = ['displayText' => 'ASC'];
break;
case 'priority':
$orderBy = ['priority' => 'ASC'];
break;
case 'priority_message':
$orderBy = ['priorityMessage' => 'ASC'];
break;
case 'option_order':
$orderBy = ['optionOrder' => 'ASC'];
break;
}
$result = Database::getManager()
->getRepository('ChamiloCoreBundle:ExtraFieldOptions')
->findBy(['field' => $field_id], $orderBy);
if (!$result) {
return false;
}
$options = [];
/** @var ExtraFieldOptions $row */
foreach ($result as $row) {
$option = [
'id' => $row->getId(),
'field_id' => $row->getField()->getId(),
'option_value' => $row->getValue(),
'display_text' => self::translateDisplayName($row->getDisplayText()),
'priority' => $row->getPriority(),
'priority_message' => $row->getPriorityMessage(),
'option_order' => $row->getOptionOrder(),
];
if ($add_id_in_array) {
$options[$row->getId()] = $option;
continue;
}
$options[] = $option;
}
return $options;
}
/**
* Get options for a specific field as array or in JSON format suited for the double-select format.
*
* @param int $option_value_id Option value ID
* @param bool $to_json Return format (whether it should be formatted to JSON or not)
*
* @return mixed Row/JSON on success
*/
public function get_second_select_field_options_by_field($option_value_id, $to_json = false)
{
$em = Database::getManager();
$option = $em->find('ChamiloCoreBundle:ExtraFieldOptions', intval($option_value_id));
if (!$option) {
return !$to_json ? [] : '{}';
}
$subOptions = $em
->getRepository('ChamiloCoreBundle:ExtraFieldOptions')
->findSecondaryOptions($option);
$optionsInfo = [];
/** @var ExtraFieldOptions $subOption */
foreach ($subOptions as $subOption) {
$optionsInfo[] = [
'id' => $subOption->getId(),
'field_id' => $subOption->getField()->getId(),
'option_value' => $subOption->getValue(),
'display_text' => $subOption->getDisplayText(),
'priority' => $subOption->getPriority(),
'priority_message' => $subOption->getPriorityMessage(),
'option_order' => $subOption->getOptionOrder(),
];
}
if (!$to_json) {
return $optionsInfo;
}
$json = [];
foreach ($optionsInfo as $optionInfo) {
$json[$optionInfo['id']] = $optionInfo['display_text'];
}
return json_encode($json);
}
/**
* Get options for a specific field as string split by ;.
*
* @param int $field_id
* @param string $ordered_by Extra query bit for reordering
*
* @return string HTML string of options
* @assert (0, '') === null
*/
public function get_field_options_by_field_to_string($field_id, $ordered_by = null)
{
$field = new ExtraField($this->type);
$field_info = $field->get($field_id);
$options = self::get_field_options_by_field($field_id, false, $ordered_by);
$elements = [];
if (!empty($options)) {
switch ($field_info['field_type']) {
case ExtraField::FIELD_TYPE_DOUBLE_SELECT:
$html = ExtraField::extra_field_double_select_convert_array_to_string($options);
break;
case ExtraField::FIELD_TYPE_SELECT_WITH_TEXT_FIELD:
$html = ExtraField::extraFieldSelectWithTextConvertArrayToString($options);
break;
case ExtraField::FIELD_TYPE_TRIPLE_SELECT:
$html = ExtraField::tripleSelectConvertArrayToString($options);
break;
default:
foreach ($options as $option) {
$elements[] = $option['option_value'];
}
$html = implode(';', $elements);
break;
}
return $html;
}
return null;
}
/**
* Get the maximum order value for a specific field.
*
* @param int $field_id
*
* @return int Current max ID + 1 (we start from 0)
* @assert (0, '') === 1
*/
public function get_max_order($field_id)
{
$field_id = intval($field_id);
$sql = "SELECT MAX(option_order)
FROM {$this->table}
WHERE field_id = $field_id";
$res = Database::query($sql);
$max = 1;
if (Database::num_rows($res) > 0) {
$row = Database::fetch_array($res);
$max = $row[0] + 1;
}
return $max;
}
/**
* Display a form with the options for the field_id given in REQUEST.
*/
public function display()
{
// action links
echo '<div class="actions">';
$field_id = isset($_REQUEST['field_id']) ? intval($_REQUEST['field_id']) : null;
echo '<a href="'.api_get_self().'?action=add&type='.$this->type.'&field_id='.$field_id.'">'.
Display::return_icon('add_user_fields.png', get_lang('Add'), '', ICON_SIZE_MEDIUM).'</a>';
echo '</div>';
echo Display::grid_html('extra_field_options');
}
/**
* @return array
*/
public function getPriorityOptions()
{
return [
'' => get_lang('SelectAnOption'),
1 => get_lang('Success'),
2 => get_lang('Info'),
3 => get_lang('Warning'),
4 => get_lang('Error'),
];
}
/**
* @param $priority
*
* @return null|string
*/
public function getPriorityMessageType($priority)
{
switch ($priority) {
case 1:
return 'success';
case 2:
return 'info';
case 3:
return 'warning';
case 4:
return 'error';
}
return null;
}
/**
* Returns an HTML form for the current field.
*
* @param string URL to send the form to (action=...)
* @param string Type of action to offer through the form (edit, usually)
*
* @return FormValidator
*/
public function return_form($url, $action)
{
$form_name = $this->type.'_field';
$form = new FormValidator($form_name, 'post', $url);
// Setting the form elements
$header = get_lang('Add');
if ($action == 'edit') {
$header = get_lang('Modify');
}
$form->addElement('header', $header);
$id = isset($_GET['id']) ? intval($_GET['id']) : '';
$form->addElement('hidden', 'id', $id);
$form->addElement('hidden', 'type', $this->type);
$form->addElement('hidden', 'field_id', $this->fieldId);
if ($action == 'edit') {
$translateUrl = api_get_path(WEB_CODE_PATH).'extrafield/translate.php?'.http_build_query([
'extra_field_option' => $id,
]);
$translateButton = Display::toolbarButton(
get_lang('TranslateThisTerm'),
$translateUrl,
'language',
'link'
);
$form->addText(
'display_text',
[get_lang('Name'), $translateButton]
);
} else {
$form->addElement('text', 'display_text', get_lang('Name'));
}
$form->addElement('text', 'option_value', get_lang('Value'));
$form->addElement('text', 'option_order', get_lang('Order'));
$form->addElement('select', 'priority', get_lang('Priority'), $this->getPriorityOptions());
$form->addElement('textarea', 'priority_message', get_lang('PriorityOfMessage'));
$defaults = [];
if ($action == 'edit') {
// Setting the defaults
$defaults = $this->get($id, false);
$form->freeze('option_value');
$form->addButtonUpdate(get_lang('Modify'));
} else {
$form->addButtonCreate(get_lang('Add'));
}
$form->setDefaults($defaults);
// Setting the rules
$form->addRule('display_text', get_lang('ThisFieldIsRequired'), 'required');
$form->addRule('option_value', get_lang('ThisFieldIsRequired'), 'required');
return $form;
}
/**
* @param string $tag
* @param int $field_id
* @param int $limit
*
* @return array
*/
public function searchByField($tag, $field_id, $limit = 10)
{
$field_id = intval($field_id);
$limit = intval($limit);
$tag = Database::escape_string($tag);
$sql = "SELECT DISTINCT id, option_display_text
FROM {$this->table}
WHERE
field_id = '".$field_id."' AND
option_value LIKE '%$tag%'
ORDER BY option_value
LIMIT 0, $limit
";
$result = Database::query($sql);
$values = [];
if (Database::num_rows($result)) {
$values = Database::store_result($result, 'ASSOC');
}
return $values;
}
/**
* @param string $tag
* @param int $field_id
* @param int $limit
*
* @return string
*/
public function getSearchOptionsByField($tag, $field_id, $limit = 10)
{
$result = $this->searchByField($tag, $field_id, $limit = 10);
$values = [];
$json = null;
if (!empty($result)) {
foreach ($result as $item) {
$values[] = [
'value' => $item['id'],
'caption' => $item['option_display_text'],
];
}
$json = json_encode($values);
}
return $json;
}
/**
* Gets an element.
*
* @param int $id
* @param bool $translateDisplayText Optional
*
* @return array
*/
public function get($id, $translateDisplayText = true)
{
$info = parent::get($id);
if ($translateDisplayText) {
$info['display_text'] = self::translateDisplayName($info['display_text']);
}
return $info;
}
/**
* Translate the display text for a extra field option.
*
* @param string $defaultDisplayText
*
* @return string
*/
public static function translateDisplayName($defaultDisplayText)
{
$variableLanguage = self::getLanguageVariable($defaultDisplayText);
return isset($GLOBALS[$variableLanguage]) ? $GLOBALS[$variableLanguage] : $defaultDisplayText;
}
/**
* @param $defaultDisplayText
*
* @return mixed|string
*/
public static function getLanguageVariable($defaultDisplayText)
{
$variableLanguage = api_replace_dangerous_char($defaultDisplayText);
$variableLanguage = str_replace('-', '_', $variableLanguage);
$variableLanguage = api_underscore_to_camel_case($variableLanguage);
return $variableLanguage;
}
/**
* @param null $options
*
* @return array
*/
public function get_all($options = null)
{
$result = parent::get_all($options);
foreach ($result as &$row) {
$row['display_text'] = self::translateDisplayName($row['display_text']);
}
return $result;
}
}