mirror of
https://github.com/Combodo/iTop.git
synced 2026-02-13 07:24:13 +01:00
* Add 'search_manual_submit' config parameter to manage auto-submit * bugfixes ** date i18n is now handled (using two new options `datepicker.dateFormat` and `datepicker.timeFormat` computed from `AttributeDateTime::GetFormat()->ToDatePicker()` ** handling of `empty` `not empty` operator titles *** it led to tohers bugfixes and a redesign of the `_computeTitle` (from overwriting to extension using ie `_computeBetweenDaysOperatorTitle` *** some bug still remain, because autocomplete needs to been finished before checking on them * Promote 'friendlyname' in the 'most popular' list * bugfixes ** filters (criterions, enum and FK without autocomplete) now ignore accent on matching with user input ** this is done by using a pre-existing tool used only by the portal, so it was moved to the core (latinize.min.js) * Integration with manual submit parameter on client side. * bugfixes : history/breadcrumb had several c[menu] appended (one for each refresh) * Fix various sanity bugs SVN:trunk[5632]
648 lines
19 KiB
PHP
648 lines
19 KiB
PHP
<?php
|
|
/**
|
|
* Copyright (C) 2010-2018 Combodo SARL
|
|
*
|
|
* This file is part of iTop.
|
|
*
|
|
* iTop is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU Affero General Public License as published by
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* iTop is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU Affero General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Affero General Public License
|
|
* along with iTop. If not, see <http://www.gnu.org/licenses/>
|
|
*
|
|
*/
|
|
|
|
/**
|
|
* Convert structures from OQL expressions into structure for the search form
|
|
*/
|
|
namespace Combodo\iTop\Application\Search\CriterionConversion;
|
|
|
|
|
|
use AttributeDate;
|
|
use AttributeDateTime;
|
|
use AttributeDefinition;
|
|
use Combodo\iTop\Application\Search\CriterionConversionAbstract;
|
|
use DateInterval;
|
|
use DateTime;
|
|
use Dict;
|
|
use Exception;
|
|
use MetaModel;
|
|
|
|
class CriterionToSearchForm extends CriterionConversionAbstract
|
|
{
|
|
|
|
/**
|
|
* @param array $aAndCriterionRaw
|
|
* @param array $aFieldsByCategory
|
|
*
|
|
* @param array $aClasses all the classes of the filter
|
|
*
|
|
* @param bool $bIsRemovable
|
|
*
|
|
* @return array
|
|
*/
|
|
public static function Convert($aAndCriterionRaw, $aFieldsByCategory, $aClasses, $bIsRemovable = true)
|
|
{
|
|
$aAllFields = array();
|
|
foreach($aFieldsByCategory as $aFields)
|
|
{
|
|
foreach($aFields as $aField)
|
|
{
|
|
$sAlias = $aField['class_alias'];
|
|
$sCode = $aField['code'];
|
|
$aAllFields["$sAlias.$sCode"] = $aField;
|
|
}
|
|
}
|
|
$aAndCriterion = array();
|
|
$aMappingOperatorToFunction = array(
|
|
AttributeDefinition::SEARCH_WIDGET_TYPE_STRING => 'TextToSearchForm',
|
|
AttributeDefinition::SEARCH_WIDGET_TYPE_EXTERNAL_FIELD => 'ExternalFieldToSearchForm',
|
|
AttributeDefinition::SEARCH_WIDGET_TYPE_DATE => 'DateTimeToSearchForm',
|
|
AttributeDefinition::SEARCH_WIDGET_TYPE_DATE_TIME => 'DateTimeToSearchForm',
|
|
AttributeDefinition::SEARCH_WIDGET_TYPE_NUMERIC => 'NumericToSearchForm',
|
|
AttributeDefinition::SEARCH_WIDGET_TYPE_EXTERNAL_KEY => 'ExternalKeyToSearchForm',
|
|
AttributeDefinition::SEARCH_WIDGET_TYPE_HIERARCHICAL_KEY => 'ExternalKeyToSearchForm',
|
|
AttributeDefinition::SEARCH_WIDGET_TYPE_ENUM => 'EnumToSearchForm',
|
|
);
|
|
|
|
foreach($aAndCriterionRaw as $aCriteria)
|
|
{
|
|
if (isset($aCriteria['label']))
|
|
{
|
|
$aCriteria['label'] = preg_replace("@\)$@", '', $aCriteria['label']);
|
|
$aCriteria['label'] = preg_replace("@^\(@", '', $aCriteria['label']);
|
|
}
|
|
$aCriteria['is_removable'] = $bIsRemovable;
|
|
|
|
$sClass = '';
|
|
if (isset($aCriteria['ref']))
|
|
{
|
|
$aRef = explode('.', $aCriteria['ref']);
|
|
if (isset($aClasses[$aRef[0]]))
|
|
{
|
|
$sClass = $aClasses[$aRef[0]];
|
|
$aCriteria['class'] = $sClass;
|
|
}
|
|
}
|
|
|
|
// Check criteria validity
|
|
if (!isset($aCriteria['ref']) || !isset($aAllFields[$aCriteria['ref']]))
|
|
{
|
|
$aCriteria['widget'] = AttributeDefinition::SEARCH_WIDGET_TYPE_RAW;
|
|
$aCriteria['label'] = Dict::S('UI:Search:Criteria:Raw:Filtered');
|
|
if (isset($aCriteria['ref']))
|
|
{
|
|
try
|
|
{
|
|
$aCriteria['label'] = Dict::Format('UI:Search:Criteria:Raw:FilteredOn', MetaModel::GetName($sClass));
|
|
}
|
|
catch (Exception $e)
|
|
{
|
|
}
|
|
}
|
|
}
|
|
if (array_key_exists('widget', $aCriteria))
|
|
{
|
|
if (array_key_exists($aCriteria['widget'], $aMappingOperatorToFunction))
|
|
{
|
|
$sFct = $aMappingOperatorToFunction[$aCriteria['widget']];
|
|
$aAndCriterion[] = self::$sFct($aCriteria, $aAllFields);
|
|
}
|
|
else
|
|
{
|
|
$aAndCriterion[] = $aCriteria;
|
|
}
|
|
}
|
|
}
|
|
|
|
// Regroup criterion by variable name (no ref first)
|
|
usort($aAndCriterion, function ($a, $b) {
|
|
if (array_key_exists('ref', $a) || array_key_exists('ref', $b))
|
|
{
|
|
if (array_key_exists('ref', $a) && array_key_exists('ref', $b))
|
|
{
|
|
$iRefCmp = strcmp($a['ref'], $b['ref']);
|
|
if ($iRefCmp != 0) return $iRefCmp;
|
|
|
|
return strcmp($a['operator'], $b['operator']);
|
|
}
|
|
if (array_key_exists('ref', $a))
|
|
{
|
|
return 1;
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
if (array_key_exists('oql', $a) && array_key_exists('oql', $b))
|
|
{
|
|
return strcmp($a['oql'], $b['oql']);
|
|
}
|
|
|
|
return 0;
|
|
});
|
|
|
|
$aMergeFctByWidget = array(
|
|
AttributeDefinition::SEARCH_WIDGET_TYPE_DATE => 'MergeDate',
|
|
AttributeDefinition::SEARCH_WIDGET_TYPE_DATE_TIME => 'MergeDateTime',
|
|
AttributeDefinition::SEARCH_WIDGET_TYPE_NUMERIC => 'MergeNumeric',
|
|
AttributeDefinition::SEARCH_WIDGET_TYPE_ENUM => 'MergeEnumExtKeys',
|
|
AttributeDefinition::SEARCH_WIDGET_TYPE_EXTERNAL_KEY => 'MergeEnumExtKeys',
|
|
);
|
|
|
|
$aPrevCriterion = null;
|
|
$aMergedCriterion = array();
|
|
foreach($aAndCriterion as $aCurrCriterion)
|
|
{
|
|
if (!is_null($aPrevCriterion))
|
|
{
|
|
if (array_key_exists('ref', $aPrevCriterion))
|
|
{
|
|
// If previous has ref, the current has ref as the array is sorted with all without ref first
|
|
if (strcmp($aPrevCriterion['ref'], $aCurrCriterion['ref']) == 0)
|
|
{
|
|
// Same attribute, try to merge
|
|
if (array_key_exists('widget', $aCurrCriterion))
|
|
{
|
|
if (array_key_exists($aCurrCriterion['widget'], $aMergeFctByWidget))
|
|
{
|
|
$sFct = $aMergeFctByWidget[$aCurrCriterion['widget']];
|
|
$aPrevCriterion = self::$sFct($aPrevCriterion, $aCurrCriterion, $aMergedCriterion);
|
|
continue;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
$aMergedCriterion[] = $aPrevCriterion;
|
|
}
|
|
|
|
$aPrevCriterion = $aCurrCriterion;
|
|
}
|
|
if (!is_null($aPrevCriterion))
|
|
{
|
|
$aMergedCriterion[] = $aPrevCriterion;
|
|
}
|
|
|
|
// Sort by label criterion by variable name (no ref first)
|
|
usort($aMergedCriterion, function ($a, $b) {
|
|
if (($a['widget'] === AttributeDefinition::SEARCH_WIDGET_TYPE_RAW) ||
|
|
($b['widget'] === AttributeDefinition::SEARCH_WIDGET_TYPE_RAW))
|
|
{
|
|
if (($a['widget'] === AttributeDefinition::SEARCH_WIDGET_TYPE_RAW) &&
|
|
($b['widget'] === AttributeDefinition::SEARCH_WIDGET_TYPE_RAW))
|
|
{
|
|
if (!isset($a['label'])) return -1;
|
|
if (!isset($b['label'])) return 1;
|
|
return strcmp($a['label'], $b['label']);
|
|
}
|
|
if ($a['widget'] === AttributeDefinition::SEARCH_WIDGET_TYPE_RAW)
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
if (!isset($a['label'])) return -1;
|
|
if (!isset($b['label'])) return 1;
|
|
return strcmp($a['label'], $b['label']);
|
|
});
|
|
|
|
return $aMergedCriterion;
|
|
}
|
|
|
|
/**
|
|
* @param $aPrevCriterion
|
|
* @param $aCurrCriterion
|
|
* @param $aMergedCriterion
|
|
*
|
|
* @return Current criteria or null if merged
|
|
* @throws \Exception
|
|
*/
|
|
protected static function MergeDate($aPrevCriterion, $aCurrCriterion, &$aMergedCriterion)
|
|
{
|
|
$sPrevOperator = $aPrevCriterion['operator'];
|
|
$sCurrOperator = $aCurrCriterion['operator'];
|
|
if ((($sPrevOperator != '<') && ($sPrevOperator != '<=')) || (($sCurrOperator != '>') && ($sCurrOperator != '>=')))
|
|
{
|
|
$aMergedCriterion[] = $aPrevCriterion;
|
|
|
|
return $aCurrCriterion;
|
|
}
|
|
|
|
// Merge into 'between' operation.
|
|
// The ends of the interval are included
|
|
$aCurrCriterion['operator'] = 'between_dates';
|
|
$oFormat = AttributeDate::GetFormat();
|
|
$sLastDate = $aPrevCriterion['values'][0]['value'];
|
|
$oDate = new DateTime($sLastDate);
|
|
if ($sPrevOperator == '<')
|
|
{
|
|
// previous day to include ends
|
|
$oDate->sub(DateInterval::createFromDateString('1 day'));
|
|
}
|
|
$sLastDateValue = $oDate->format(AttributeDate::GetSQLFormat());
|
|
$sLastDateLabel = $oFormat->format($oDate);
|
|
|
|
$sFirstDate = $aCurrCriterion['values'][0]['value'];
|
|
$oDate = new DateTime($sFirstDate);
|
|
if ($sCurrOperator == '>')
|
|
{
|
|
// next day to include ends
|
|
$oDate->add(DateInterval::createFromDateString('1 day'));
|
|
}
|
|
$sFirstDateValue = $oDate->format(AttributeDate::GetSQLFormat());
|
|
$sFirstDateLabel = $oFormat->format($oDate);
|
|
|
|
$aCurrCriterion['values'] = array();
|
|
$aCurrCriterion['values'][] = array('value' => $sFirstDateValue, 'label' => $sFirstDateLabel);
|
|
$aCurrCriterion['values'][] = array('value' => $sLastDateValue, 'label' => $sLastDateLabel);
|
|
|
|
$aCurrCriterion['oql'] = "({$aPrevCriterion['oql']} AND {$aCurrCriterion['oql']})";
|
|
$aCurrCriterion['label'] = $aPrevCriterion['label'].' '.Dict::S('Expression:Operator:AND', 'AND').' '.$aCurrCriterion['label'];
|
|
|
|
$aMergedCriterion[] = $aCurrCriterion;
|
|
|
|
return null;
|
|
}
|
|
|
|
/**
|
|
* @param $aPrevCriterion
|
|
* @param $aCurrCriterion
|
|
* @param $aMergedCriterion
|
|
*
|
|
* @return Current criteria or null if merged
|
|
* @throws \Exception
|
|
*/
|
|
protected static function MergeDateTime($aPrevCriterion, $aCurrCriterion, &$aMergedCriterion)
|
|
{
|
|
$sPrevOperator = $aPrevCriterion['operator'];
|
|
$sCurrOperator = $aCurrCriterion['operator'];
|
|
if ((($sPrevOperator != '<') && ($sPrevOperator != '<=')) || (($sCurrOperator != '>') && ($sCurrOperator != '>=')))
|
|
{
|
|
$aMergedCriterion[] = $aPrevCriterion;
|
|
|
|
return $aCurrCriterion;
|
|
}
|
|
|
|
// Merge into 'between' operation.
|
|
// The ends of the interval are included
|
|
$sLastDate = $aPrevCriterion['values'][0]['value'];
|
|
$sFirstDate = $aCurrCriterion['values'][0]['value'];
|
|
$oDate = new DateTime($sLastDate);
|
|
$aCurrCriterion['operator'] = 'between_dates';
|
|
$sInterval = '1 second';
|
|
|
|
if ($sPrevOperator == '<')
|
|
{
|
|
// previous day/second to include ends
|
|
$oDate->sub(DateInterval::createFromDateString($sInterval));
|
|
}
|
|
$sLastDateValue = $oDate->format(AttributeDateTime::GetSQLFormat());
|
|
$sLastDateLabel = AttributeDateTime::GetFormat()->Format($sLastDateValue);
|
|
|
|
$oDate = new DateTime($sFirstDate);
|
|
if ($sCurrOperator == '>')
|
|
{
|
|
// next day/second to include ends
|
|
$oDate->add(DateInterval::createFromDateString($sInterval));
|
|
}
|
|
$sFirstDateValue = $oDate->format(AttributeDateTime::GetSQLFormat());
|
|
$sFirstDateLabel = AttributeDateTime::GetFormat()->Format($sFirstDateValue);
|
|
|
|
$aCurrCriterion['values'] = array();
|
|
$aCurrCriterion['values'][] = array('value' => $sFirstDateValue, 'label' => $sFirstDateLabel);
|
|
$aCurrCriterion['values'][] = array('value' => $sLastDateValue, 'label' => $sLastDateLabel);
|
|
|
|
$aCurrCriterion['oql'] = "({$aPrevCriterion['oql']} AND {$aCurrCriterion['oql']})";
|
|
$aCurrCriterion['label'] = $aPrevCriterion['label'].' '.Dict::S('Expression:Operator:AND',
|
|
'AND').' '.$aCurrCriterion['label'];
|
|
|
|
$aMergedCriterion[] = $aCurrCriterion;
|
|
|
|
return null;
|
|
}
|
|
|
|
/**
|
|
* @param $aPrevCriterion
|
|
* @param $aCurrCriterion
|
|
* @param $aMergedCriterion
|
|
*
|
|
* @return Current criteria or null if merged
|
|
* @throws \Exception
|
|
*/
|
|
protected static function MergeNumeric($aPrevCriterion, $aCurrCriterion, &$aMergedCriterion)
|
|
{
|
|
$sPrevOperator = $aPrevCriterion['operator'];
|
|
$sCurrOperator = $aCurrCriterion['operator'];
|
|
if (($sPrevOperator != '<=') || ($sCurrOperator != '>='))
|
|
{
|
|
$aMergedCriterion[] = $aPrevCriterion;
|
|
|
|
return $aCurrCriterion;
|
|
}
|
|
|
|
// Merge into 'between' operation.
|
|
$sLastNum = $aPrevCriterion['values'][0]['value'];
|
|
$sFirstNum = $aCurrCriterion['values'][0]['value'];
|
|
$aCurrCriterion['values'] = array();
|
|
$aCurrCriterion['values'][] = array('value' => $sFirstNum, 'label' => "$sFirstNum");
|
|
$aCurrCriterion['values'][] = array('value' => $sLastNum, 'label' => "$sLastNum");
|
|
|
|
$aCurrCriterion['oql'] = "({$aPrevCriterion['oql']} AND {$aCurrCriterion['oql']})";
|
|
$aCurrCriterion['label'] = $aPrevCriterion['label'].' '.Dict::S('Expression:Operator:AND', 'AND').' '.$aCurrCriterion['label'];
|
|
$aCurrCriterion['operator'] = 'between';
|
|
|
|
$aMergedCriterion[] = $aCurrCriterion;
|
|
|
|
return null;
|
|
}
|
|
|
|
private static function SerializeValues($aValues)
|
|
{
|
|
$aSerializedValues = array();
|
|
foreach($aValues as $aValue)
|
|
{
|
|
$aSerializedValues[] = serialize($aValue);
|
|
}
|
|
|
|
return $aSerializedValues;
|
|
}
|
|
|
|
protected static function MergeEnumExtKeys($aPrevCriterion, $aCurrCriterion, &$aMergedCriterion)
|
|
{
|
|
$aFirstValues = self::SerializeValues($aPrevCriterion['values']);
|
|
$aNextValues = self::SerializeValues($aCurrCriterion['values']);
|
|
|
|
// Keep only the common values
|
|
$aCurrCriterion['values'] = array_map("unserialize", array_intersect($aFirstValues, $aNextValues));
|
|
|
|
$aMergedCriterion[] = $aCurrCriterion;
|
|
return null;
|
|
}
|
|
|
|
protected static function TextToSearchForm($aCriteria, $aFields)
|
|
{
|
|
$sOperator = $aCriteria['operator'];
|
|
$sValue = $aCriteria['values'][0]['value'];
|
|
|
|
$bStartWithPercent = substr($sValue, 0, 1) == '%' ? true : false;
|
|
$bEndWithPercent = substr($sValue, -1) == '%' ? true : false;
|
|
|
|
switch (true)
|
|
{
|
|
case ('' == $sValue and ($sOperator == '=' or $sOperator == 'LIKE')):
|
|
$aCriteria['operator'] = CriterionConversionAbstract::OP_EMPTY;
|
|
break;
|
|
case ('' == $sValue and $sOperator == '!='):
|
|
$aCriteria['operator'] = CriterionConversionAbstract::OP_NOT_EMPTY;
|
|
break;
|
|
case ($sOperator == 'LIKE' && $bStartWithPercent && $bEndWithPercent):
|
|
$aCriteria['operator'] = CriterionConversionAbstract::OP_CONTAINS;
|
|
$sValue = substr($sValue, 1, -1);
|
|
$aCriteria['values'][0]['value'] = $sValue;
|
|
$aCriteria['values'][0]['label'] = "$sValue";
|
|
break;
|
|
case ($sOperator == 'LIKE' && $bStartWithPercent):
|
|
$aCriteria['operator'] = CriterionConversionAbstract::OP_ENDS_WITH;
|
|
$sValue = substr($sValue, 1);
|
|
$aCriteria['values'][0]['value'] = $sValue;
|
|
$aCriteria['values'][0]['label'] = "$sValue";
|
|
break;
|
|
case ($sOperator == 'LIKE' && $bEndWithPercent):
|
|
$aCriteria['operator'] = CriterionConversionAbstract::OP_STARTS_WITH;
|
|
$sValue = substr($sValue, 0, -1);
|
|
$aCriteria['values'][0]['value'] = $sValue;
|
|
$aCriteria['values'][0]['label'] = "$sValue";
|
|
break;
|
|
}
|
|
|
|
return $aCriteria;
|
|
}
|
|
|
|
protected static function ExternalFieldToSearchForm($aCriteria, $aFields)
|
|
{
|
|
$sOperator = $aCriteria['operator'];
|
|
$sValue = $aCriteria['values'][0]['value'];
|
|
|
|
$bStartWithPercent = substr($sValue, 0, 1) == '%' ? true : false;
|
|
$bEndWithPercent = substr($sValue, -1) == '%' ? true : false;
|
|
|
|
switch (true)
|
|
{
|
|
case ($sOperator == 'ISNULL'):
|
|
case ('' == $sValue and ($sOperator == 'LIKE')):
|
|
$aCriteria['operator'] = CriterionConversionAbstract::OP_EMPTY;
|
|
break;
|
|
case ('' == $sValue and $sOperator == '!='):
|
|
$aCriteria['operator'] = CriterionConversionAbstract::OP_NOT_EMPTY;
|
|
break;
|
|
case ($sOperator == 'LIKE' && $bStartWithPercent && $bEndWithPercent):
|
|
$aCriteria['operator'] = CriterionConversionAbstract::OP_CONTAINS;
|
|
$sValue = substr($sValue, 1, -1);
|
|
$aCriteria['values'][0]['value'] = $sValue;
|
|
$aCriteria['values'][0]['label'] = "$sValue";
|
|
break;
|
|
case ($sOperator == 'LIKE' && $bStartWithPercent):
|
|
$aCriteria['operator'] = CriterionConversionAbstract::OP_ENDS_WITH;
|
|
$sValue = substr($sValue, 1);
|
|
$aCriteria['values'][0]['value'] = $sValue;
|
|
$aCriteria['values'][0]['label'] = "$sValue";
|
|
break;
|
|
case ($sOperator == 'LIKE' && $bEndWithPercent):
|
|
$aCriteria['operator'] = CriterionConversionAbstract::OP_STARTS_WITH;
|
|
$sValue = substr($sValue, 0, -1);
|
|
$aCriteria['values'][0]['value'] = $sValue;
|
|
$aCriteria['values'][0]['label'] = "$sValue";
|
|
break;
|
|
}
|
|
|
|
return $aCriteria;
|
|
}
|
|
|
|
protected static function DateTimeToSearchForm($aCriteria, $aFields)
|
|
{
|
|
if (!array_key_exists('is_relative', $aCriteria) || !$aCriteria['is_relative'])
|
|
{
|
|
// Convert '=' in 'between'
|
|
if (isset($aCriteria['operator']) && ($aCriteria['operator'] === '='))
|
|
{
|
|
$aCriteria['operator'] = CriterionConversionAbstract::OP_BETWEEN_DATES;
|
|
$sWidget = $aCriteria['widget'];
|
|
if ($sWidget == AttributeDefinition::SEARCH_WIDGET_TYPE_DATE)
|
|
{
|
|
$aCriteria['values'][1] = $aCriteria['values'][0];
|
|
}
|
|
else
|
|
{
|
|
$sDate = $aCriteria['values'][0]['value'];
|
|
$oDate = new DateTime($sDate);
|
|
|
|
$sFirstDateValue = $oDate->format(AttributeDateTime::GetSQLFormat());
|
|
$sFirstDateLabel = AttributeDateTime::GetFormat()->Format($sFirstDateValue);
|
|
$aCriteria['values'][0] = array('value' => $sFirstDateValue, 'label' => "$sFirstDateLabel");
|
|
|
|
|
|
$oDate->add(DateInterval::createFromDateString('1 day'));
|
|
$oDate->sub(DateInterval::createFromDateString('1 second'));
|
|
|
|
$sLastDateValue = $oDate->format(AttributeDateTime::GetSQLFormat());
|
|
$sLastDateLabel = AttributeDateTime::GetFormat()->Format($sLastDateValue);
|
|
$aCriteria['values'][1] = array('value' => $sLastDateValue, 'label' => "$sLastDateLabel");
|
|
}
|
|
}
|
|
|
|
return $aCriteria;
|
|
}
|
|
|
|
if (isset($aCriteria['values'][0]['value']))
|
|
{
|
|
$sLabel = $aCriteria['values'][0]['value'];
|
|
if (isset($aCriteria['verb']))
|
|
{
|
|
switch ($aCriteria['verb'])
|
|
{
|
|
case 'DATE_SUB':
|
|
$sLabel = '-'.$sLabel;
|
|
break;
|
|
case 'DATE_ADD':
|
|
$sLabel = '+'.$sLabel;
|
|
break;
|
|
}
|
|
}
|
|
if (isset($aCriteria['unit']))
|
|
{
|
|
$sLabel .= Dict::S('Expression:Unit:Short:'.$aCriteria['unit'], $aCriteria['unit']);
|
|
}
|
|
$aCriteria['values'][0]['label'] = "$sLabel";
|
|
}
|
|
|
|
// Temporary until the JS widget support relative dates
|
|
$aCriteria['widget'] = AttributeDefinition::SEARCH_WIDGET_TYPE_RAW;
|
|
|
|
return $aCriteria;
|
|
}
|
|
|
|
protected static function NumericToSearchForm($aCriteria, $aFields)
|
|
{
|
|
if ($aCriteria['operator'] == 'ISNULL')
|
|
{
|
|
$aCriteria['operator'] = CriterionConversionAbstract::OP_EMPTY;
|
|
}
|
|
|
|
return $aCriteria;
|
|
}
|
|
|
|
protected static function EnumToSearchForm($aCriteria, $aFields)
|
|
{
|
|
$sOperator = $aCriteria['operator'];
|
|
switch ($sOperator)
|
|
{
|
|
case '=':
|
|
// Same as IN
|
|
$aCriteria['operator'] = CriterionConversionAbstract::OP_IN;
|
|
break;
|
|
case 'NOT IN':
|
|
case 'NOTIN':
|
|
case '!=':
|
|
// Same as NOT IN
|
|
$aCriteria = self::RevertValues($aCriteria, $aFields);
|
|
break;
|
|
case 'IN':
|
|
// Nothing special to do
|
|
break;
|
|
case 'OR':
|
|
case 'ISNULL':
|
|
// Special case when undefined and/or other values are selected
|
|
$aCriteria['operator'] = CriterionConversionAbstract::OP_IN;
|
|
if (isset($aCriteria['has_undefined']) && $aCriteria['has_undefined'])
|
|
{
|
|
if (!isset($aCriteria['values']))
|
|
{
|
|
$aCriteria['values'] = array();
|
|
}
|
|
// Convention for 'undefined' enums
|
|
$aCriteria['values'][] = array('value' => 'null', 'label' => Dict::S('Enum:Undefined'));
|
|
}
|
|
break;
|
|
default:
|
|
// Unknown operator
|
|
$aCriteria['widget'] = AttributeDefinition::SEARCH_WIDGET_TYPE_RAW;
|
|
break;
|
|
}
|
|
|
|
return $aCriteria;
|
|
}
|
|
|
|
protected static function ExternalKeyToSearchForm($aCriteria, $aFields)
|
|
{
|
|
$sOperator = $aCriteria['operator'];
|
|
switch ($sOperator)
|
|
{
|
|
case '=':
|
|
// Same as IN
|
|
$aCriteria['operator'] = CriterionConversionAbstract::OP_IN;
|
|
break;
|
|
case 'IN':
|
|
// Nothing special to do
|
|
break;
|
|
default:
|
|
// Unknown operator
|
|
$aCriteria['widget'] = AttributeDefinition::SEARCH_WIDGET_TYPE_RAW;
|
|
break;
|
|
}
|
|
|
|
return $aCriteria;
|
|
}
|
|
|
|
/**
|
|
* @param $aCriteria
|
|
* @param $aFields
|
|
*
|
|
* @return mixed
|
|
*/
|
|
protected static function RevertValues($aCriteria, $aFields)
|
|
{
|
|
$sRef = $aCriteria['ref'];
|
|
$aValues = $aCriteria['values'];
|
|
if (array_key_exists($sRef, $aFields))
|
|
{
|
|
$aField = $aFields[$sRef];
|
|
if (array_key_exists('allowed_values', $aField) && array_key_exists('values', $aField['allowed_values']))
|
|
{
|
|
$aAllowedValues = $aField['allowed_values']['values'];
|
|
}
|
|
else
|
|
{
|
|
// Can't obtain the list of allowed values, just set as unknown
|
|
$aCriteria['widget'] = AttributeDefinition::SEARCH_WIDGET_TYPE_RAW;
|
|
}
|
|
}
|
|
|
|
if (isset($aAllowedValues))
|
|
{
|
|
foreach($aValues as $aValue)
|
|
{
|
|
$sValue = $aValue['value'];
|
|
unset($aAllowedValues[$sValue]);
|
|
}
|
|
$aCriteria['values'] = array();
|
|
|
|
foreach($aAllowedValues as $sValue => $sLabel)
|
|
{
|
|
$aValue = array('value' => $sValue, 'label' => "$sLabel");
|
|
$aCriteria['values'][] = $aValue;
|
|
}
|
|
$aCriteria['operator'] = 'IN';
|
|
}
|
|
|
|
return $aCriteria;
|
|
}
|
|
|
|
} |