diff --git a/application/cmdbabstract.class.inc.php b/application/cmdbabstract.class.inc.php index 26f31e2f2..c114942bb 100644 --- a/application/cmdbabstract.class.inc.php +++ b/application/cmdbabstract.class.inc.php @@ -17,9 +17,11 @@ * You should have received a copy of the GNU Affero General Public License */ +use Combodo\iTop\Application\Search\SearchForm; use Combodo\iTop\Application\UI\Component\Alert\AlertFactory; use Combodo\iTop\Application\UI\Component\Button\Button; use Combodo\iTop\Application\UI\Component\Button\ButtonFactory; +use Combodo\iTop\Application\UI\Component\DataTable\DataTableSettings; use Combodo\iTop\Application\UI\Component\Field\Field; use Combodo\iTop\Application\UI\Component\FieldSet\FieldSet; use Combodo\iTop\Application\UI\Component\Form\Form; @@ -50,13 +52,15 @@ require_once(APPROOT.'application/ui.linksdirectwidget.class.inc.php'); require_once(APPROOT.'application/ui.passwordwidget.class.inc.php'); require_once(APPROOT.'application/ui.extkeywidget.class.inc.php'); require_once(APPROOT.'application/ui.htmleditorwidget.class.inc.php'); -require_once(APPROOT.'application/datatable.class.inc.php'); require_once(APPROOT.'sources/application/search/searchform.class.inc.php'); require_once(APPROOT.'sources/application/search/criterionparser.class.inc.php'); require_once(APPROOT.'sources/application/search/criterionconversionabstract.class.inc.php'); require_once(APPROOT.'sources/application/search/criterionconversion/criteriontooql.class.inc.php'); require_once(APPROOT.'sources/application/search/criterionconversion/criteriontosearchform.class.inc.php'); +use Combodo\iTop\Application\UI\Component\DataTable\DataTableFactory; +use Combodo\iTop\Renderer\Console\ConsoleFormRenderer; + /** * Class cmdbAbstractObject */ @@ -389,11 +393,11 @@ EOF $aIcons[] = "
  $sLabel
"; } - if (count($aIcons) > 0) { - $sTags = '
'.implode(' ', $aIcons).'
'; - } else { - $sTags = ''; - } +// if (count($aIcons) > 0) { +// $sTags = '
'.implode(' ', $aIcons).'
'; +// } else { +// $sTags = ''; +// } $oPage->AddUiBlock(TitleFactory::MakeForObjectDetails($this)); } @@ -1042,7 +1046,7 @@ HTML */ public static function DisplaySet(WebPage $oPage, CMDBObjectSet $oSet, $aExtraParams = array()) { - $oPage->add(self::GetDisplaySet($oPage, $oSet, $aExtraParams)); + $oPage->AddUiBlock(self::GetDisplaySetBlock($oPage, $oSet, $aExtraParams)); } /** @@ -1100,14 +1104,16 @@ HTML * * @throws \CoreException*@throws \Exception * @throws \ApplicationException + * @deprecated 3.0.0 use GetDisplaySetBlock */ public static function GetDisplaySet(WebPage $oPage, CMDBObjectSet $oSet, $aExtraParams = array()) { - if ($oPage->IsPrintableVersion() || $oPage->is_pdf()) - { - return self::GetDisplaySetForPrinting($oPage, $oSet, $aExtraParams); + $oPage->AddUiBlock(static::GetDisplaySetBlock( $oPage, $oSet, $aExtraParams )); + return ""; } + public static function GetDisplaySetBlock(WebPage $oPage, CMDBObjectSet $oSet, $aExtraParams = array()) + { if (empty($aExtraParams['currentId'])) { $iListId = $oPage->GetUniqueId(); // Works only if not in an Ajax page !! @@ -1117,136 +1123,8 @@ HTML $iListId = $aExtraParams['currentId']; } - // Initialize and check the parameters - $bViewLink = isset($aExtraParams['view_link']) ? $aExtraParams['view_link'] : true; - $sLinkageAttribute = isset($aExtraParams['link_attr']) ? $aExtraParams['link_attr'] : ''; - $iLinkedObjectId = isset($aExtraParams['object_id']) ? $aExtraParams['object_id'] : 0; - $sTargetAttr = isset($aExtraParams['target_attr']) ? $aExtraParams['target_attr'] : ''; - if (!empty($sLinkageAttribute)) - { - if ($iLinkedObjectId == 0) - { - // if 'links' mode is requested the id of the object to link to must be specified - throw new ApplicationException(Dict::S('UI:Error:MandatoryTemplateParameter_object_id')); - } - if ($sTargetAttr == '') - { - // if 'links' mode is requested the d of the object to link to must be specified - throw new ApplicationException(Dict::S('UI:Error:MandatoryTemplateParameter_target_attr')); - } - } - $bDisplayMenu = isset($aExtraParams['menu']) ? $aExtraParams['menu'] == true : true; - $bSelectMode = isset($aExtraParams['selection_mode']) ? $aExtraParams['selection_mode'] == true : false; - $bSingleSelectMode = isset($aExtraParams['selection_type']) ? ($aExtraParams['selection_type'] == 'single') : false; - - $aExtraFieldsRaw = isset($aExtraParams['extra_fields']) ? explode(',', - trim($aExtraParams['extra_fields'])) : array(); - $aExtraFields = array(); - foreach($aExtraFieldsRaw as $sFieldName) - { - // Ignore attributes not of the main queried class - if (preg_match('/^(.*)\.(.*)$/', $sFieldName, $aMatches)) - { - $sClassAlias = $aMatches[1]; - $sAttCode = $aMatches[2]; - if ($sClassAlias == $oSet->GetFilter()->GetClassAlias()) - { - $aExtraFields[] = $sAttCode; - } - } - else - { - $aExtraFields[] = $sFieldName; - } - } - $sClassName = $oSet->GetFilter()->GetClass(); - $sZListName = isset($aExtraParams['zlist']) ? ($aExtraParams['zlist']) : 'list'; - if ($sZListName !== false) - { - $aList = self::FlattenZList(MetaModel::GetZListItems($sClassName, $sZListName)); - $aList = array_merge($aList, $aExtraFields); - } - else - { - $aList = $aExtraFields; - } - - // Filter the list to removed linked set since we are not able to display them here - foreach($aList as $index => $sAttCode) - { - $oAttDef = MetaModel::GetAttributeDef($sClassName, $sAttCode); - if ($oAttDef instanceof AttributeLinkedSet) - { - // Removed from the display list - unset($aList[$index]); - } - } - - - if (!empty($sLinkageAttribute)) - { - // The set to display is in fact a set of links between the object specified in the $sLinkageAttribute - // and other objects... - // The display will then group all the attributes related to the link itself: - // | Link_attr1 | link_attr2 | ... || Object_attr1 | Object_attr2 | Object_attr3 | .. | Object_attr_n | - $aDisplayList = array(); - $aAttDefs = MetaModel::ListAttributeDefs($sClassName); - assert(isset($aAttDefs[$sLinkageAttribute])); - $oAttDef = $aAttDefs[$sLinkageAttribute]; - assert($oAttDef->IsExternalKey()); - // First display all the attributes specific to the link record - foreach($aList as $sLinkAttCode) - { - $oLinkAttDef = $aAttDefs[$sLinkAttCode]; - if ((!$oLinkAttDef->IsExternalKey()) && (!$oLinkAttDef->IsExternalField())) - { - $aDisplayList[] = $sLinkAttCode; - } - } - // Then display all the attributes neither specific to the link record nor to the 'linkage' object (because the latter are constant) - foreach($aList as $sLinkAttCode) - { - $oLinkAttDef = $aAttDefs[$sLinkAttCode]; - if (($oLinkAttDef->IsExternalKey() && ($sLinkAttCode != $sLinkageAttribute)) - || ($oLinkAttDef->IsExternalField() && ($oLinkAttDef->GetKeyAttCode() != $sLinkageAttribute))) - { - $aDisplayList[] = $sLinkAttCode; - } - } - // First display all the attributes specific to the link - // Then display all the attributes linked to the other end of the relationship - $aList = $aDisplayList; - } - - $sSelectMode = 'none'; - if ($bSelectMode) - { - $sSelectMode = $bSingleSelectMode ? 'single' : 'multiple'; - } - - $sClassAlias = $oSet->GetClassAlias(); - $bDisplayLimit = isset($aExtraParams['display_limit']) ? $aExtraParams['display_limit'] : true; - - $sTableId = isset($aExtraParams['table_id']) ? $aExtraParams['table_id'] : null; - $aClassAliases = array($sClassAlias => $sClassName); - $oDataTable = new DataTable($iListId, $oSet, $aClassAliases, $sTableId); - $oSettings = DataTableSettings::GetDataModelSettings($aClassAliases, $bViewLink, array($sClassAlias => $aList)); - - if ($bDisplayLimit) - { - $iDefaultPageSize = appUserPreferences::GetPref('default_page_size', - MetaModel::GetConfig()->GetMinDisplayLimit()); - $oSettings->iDefaultPageSize = $iDefaultPageSize; - } - else - { - $oSettings->iDefaultPageSize = 0; - } - $oSettings->aSortOrder = MetaModel::GetOrderByDefault($sClassName); - - return $oDataTable->Display($oPage, $oSettings, $bDisplayMenu, $sSelectMode, $bViewLink, $aExtraParams); + return DataTableFactory::MakeForResult($oPage, $iListId, $oSet, $aExtraParams); } - /** * @param \WebPage $oPage * @param \CMDBObjectSet $oSet @@ -1265,6 +1143,7 @@ HTML * @throws \MissingQueryArgument * @throws \MySQLException * @throws \MySQLHasGoneAwayException + * @deprecated since 3.0.0 */ public static function GetDisplayExtendedSet(WebPage $oPage, CMDBObjectSet $oSet, $aExtraParams = array()) { @@ -1844,7 +1723,7 @@ HTML */ public static function GetSearchForm(WebPage $oPage, CMDBObjectSet $oSet, $aExtraParams = array()) { - $oSearchForm = new \Combodo\iTop\Application\Search\SearchForm(); + $oSearchForm = new SearchForm(); return $oSearchForm->GetSearchForm($oPage, $oSet, $aExtraParams); } @@ -2266,7 +2145,7 @@ EOF $sHTMLValue .= "\n"; $oForm = $value->GetForm($sFormPrefix); - $oRenderer = new \Combodo\iTop\Renderer\Console\ConsoleFormRenderer($oForm); + $oRenderer = new ConsoleFormRenderer($oForm); $aRenderRes = $oRenderer->Render(); $aFieldSetOptions = array( @@ -2537,8 +2416,7 @@ JS if (isset($aExtraParams['wizard_container']) && $aExtraParams['wizard_container']) { $sClassLabel = MetaModel::GetName($sClass); - $sHeaderTitle = Dict::Format('UI:ModificationTitle_Class_Object', $sClassLabel, - $this->GetName()); + //$sHeaderTitle = Dict::Format('UI:ModificationTitle_Class_Object', $sClassLabel, $this->GetName()); $oPage->set_title(Dict::Format('UI:ModificationPageTitle_Object_Class', $this->GetRawName(), $sClassLabel)); // Set title will take care of the encoding @@ -2615,6 +2493,7 @@ CSS $oForm->AddSubBlock(InputFactory::MakeForHidden('id', $iKey, "{$sPrefix}_id")); } $oForm->AddSubBlock(InputFactory::MakeForHidden('operation', $sOperation)); + $oForm->AddSubBlock(InputFactory::MakeForHidden('class', $sClass)); $oCancelButton = ButtonFactory::MakeForSecondaryAction(Dict::S('UI:Button:Cancel')); $oCancelButton->AddCSSClasses('action cancel'); @@ -2648,6 +2527,7 @@ CSS $sButtonsPosition = MetaModel::GetConfig()->Get('buttons_position'); $iTransactionId = isset($aExtraParams['transaction_id']) ? $aExtraParams['transaction_id'] : utils::GetNewTransactionId(); $oPage->SetTransactionId($iTransactionId); + $oForm->AddSubBlock(InputFactory::MakeForHidden('transaction_id', $iTransactionId)); $sStatesSelection = ''; if (!isset($aExtraParams['custom_operation']) && $this->IsNew()) { @@ -4315,12 +4195,9 @@ EOF if ((!$bEditMode) || ($iFlags & (OPT_ATT_READONLY | OPT_ATT_SLAVE))) { // Check if the attribute is not read-only because of a synchro... - $sSynchroIcon = ''; if ($iFlags & OPT_ATT_SLAVE) { $aReasons = array(); - $iSynchroFlags = $this->GetSynchroReplicaFlags($sAttCode, $aReasons); - $sSynchroIcon = " "; $sTip = ''; foreach($aReasons as $aRow) { @@ -4339,7 +4216,6 @@ EOF $sHTMLValue .= ''; $aFieldsMap[$sAttCode] = $sInputId; - $sComment .= $sSynchroIcon; } else { @@ -4388,7 +4264,6 @@ HTML public function GetExpectedAttributes($sCurrentState, $sStimulus, $bOnlyNewOnes) { $aTransitions = $this->EnumTransitions(); - $aStimuli = MetaModel::EnumStimuli(get_class($this)); if (!isset($aTransitions[$sStimulus])) { // Invalid stimulus @@ -4709,6 +4584,7 @@ EOF */ public static function DoBulkModify($oP, $sClass, $aSelectedObj, $sCustomOperation, $bPreview, $sCancelUrl, $aContextData = array()) { + /** @var string[] $aHeaders */ $aHeaders = array( 'form::select' => array( 'label' => "", @@ -4745,7 +4621,7 @@ EOF $iLoopTimeLimit = MetaModel::GetConfig()->Get('max_execution_time_per_loop'); foreach($aSelectedObj as $iId) { - set_time_limit($iLoopTimeLimit); + set_time_limit(intval($iLoopTimeLimit)); /** @var \cmdbAbstractObject $oObj */ $oObj = MetaModel::GetObject($sClass, $iId); $aErrors = $oObj->UpdateObjectFromPostedForm(''); @@ -4777,14 +4653,13 @@ EOF $oObj->DBUpdate(); } } - set_time_limit($iPreviousTimeLimit); + set_time_limit(intval($iPreviousTimeLimit)); $oP->Table($aHeaders, $aRows); if ($bPreview) { $sFormAction = utils::GetAbsoluteUrlAppRoot().'pages/UI.php'; // No parameter in the URL, the only parameter will be the ones passed through the form // Form to submit: $oP->add("
\n"); - $aDefaults = utils::ReadParam('default', array()); $oAppContext = new ApplicationContext(); $oP->add($oAppContext->GetForForm()); foreach($aContextData as $sKey => $value) @@ -5000,7 +4875,6 @@ EOF if (count($aObjects) == 1) { $oObj = $aObjects[0]; - $id = $oObj->GetKey(); $oP->p('

'.Dict::Format('UI:Delect:Confirm_Object', $oObj->GetHyperLink()).'

'); } else diff --git a/application/dashboard.class.inc.php b/application/dashboard.class.inc.php index 3a0428054..3539baa8d 100644 --- a/application/dashboard.class.inc.php +++ b/application/dashboard.class.inc.php @@ -20,6 +20,8 @@ use Combodo\iTop\Application\UI\Component\Button\ButtonFactory; use Combodo\iTop\Application\UI\Component\Toolbar\Toolbar; +use Combodo\iTop\Application\UI\Component\DataTable\DataTableSettings; + require_once(APPROOT.'application/dashboardlayout.class.inc.php'); require_once(APPROOT.'application/dashlet.class.inc.php'); require_once(APPROOT.'core/modelreflection.class.inc.php'); diff --git a/application/datatable.class.inc.php b/application/datatable.class.inc.php index a975e3ad3..0765c4190 100644 --- a/application/datatable.class.inc.php +++ b/application/datatable.class.inc.php @@ -28,7 +28,7 @@ class DataTable protected $sTableId; // identifier for saving the settings (combined with the class aliases) protected $oSet; // The set of objects to display protected $aClassAliases; // The aliases (alias => class) inside the set - protected $iNbObjects; // Total number of objects inthe set + protected $iNbObjects; // Total number of objects in the set protected $bUseCustomSettings; // Whether or not the current display uses custom settings protected $oDefaultSettings; // the default settings for displaying such a list protected $bShowObsoleteData; @@ -211,7 +211,7 @@ class DataTable return $sHtml; } - + /** * When refreshing the body of a paginated table, get the rows of the table (inside the TBODY) * return string The HTML rows to insert inside the node @@ -525,6 +525,7 @@ EOF; return $aAttribs; } + /** * @param $aColumns * @param $sSelectMode @@ -872,396 +873,3 @@ class PrintableDataTable extends DataTable return $sHtml; } } - -class DataTableSettings implements Serializable -{ - public $aClassAliases; - public $sTableId; - public $iDefaultPageSize; - public $aColumns; - - - /** - * DataTableSettings constructor. - * - * @param $aClassAliases - * @param null $sTableId - */ - public function __construct($aClassAliases, $sTableId = null) - { - $this->aClassAliases = $aClassAliases; - $this->sTableId = $sTableId; - $this->iDefaultPageSize = 10; - $this->aColumns = array(); - } - - /** - * @param $iDefaultPageSize - * @param $aSortOrder - * @param $aColumns - */ - protected function Init($iDefaultPageSize, $aSortOrder, $aColumns) - { - $this->iDefaultPageSize = $iDefaultPageSize; - $this->aColumns = $aColumns; - $this->FixVisibleColumns(); - } - - /** - * @return string - */ - public function serialize() - { - // Save only the 'visible' columns - $aColumns = array(); - foreach($this->aClassAliases as $sAlias => $sClass) - { - $aColumns[$sAlias] = array(); - foreach($this->aColumns[$sAlias] as $sAttCode => $aData) - { - unset($aData['label']); // Don't save the display name - unset($aData['alias']); // Don't save the alias (redundant) - unset($aData['code']); // Don't save the code (redundant) - if ($aData['checked']) - { - $aColumns[$sAlias][$sAttCode] = $aData; - } - } - } - return serialize( - array( - 'iDefaultPageSize' => $this->iDefaultPageSize, - 'aColumns' => $aColumns, - ) - ); - } - - /** - * @param string $sData - * - * @throws \Exception - */ - public function unserialize($sData) - { - $aData = unserialize($sData); - $this->iDefaultPageSize = $aData['iDefaultPageSize']; - $this->aColumns = $aData['aColumns']; - foreach($this->aClassAliases as $sAlias => $sClass) - { - foreach($this->aColumns[$sAlias] as $sAttCode => $aData) - { - $aFieldData = false; - if ($sAttCode == '_key_') - { - $aFieldData = $this->GetFieldData($sAlias, $sAttCode, null, true /* bChecked */, $aData['sort']); - } - else if (MetaModel::isValidAttCode($sClass, $sAttCode)) - { - $oAttDef = MetaModel::GetAttributeDef($sClass, $sAttCode); - $aFieldData = $this->GetFieldData($sAlias, $sAttCode, $oAttDef, true /* bChecked */, $aData['sort']); - } - - if ($aFieldData) - { - $this->aColumns[$sAlias][$sAttCode] = $aFieldData; - } - else - { - unset($this->aColumns[$sAlias][$sAttCode]); - } - } - } - $this->FixVisibleColumns(); - } - - /** - * @param $aClassAliases - * @param $bViewLink - * @param $aDefaultLists - * - * @return \DataTableSettings - * @throws \CoreException - * @throws \DictExceptionMissingString - */ - public static function GetDataModelSettings($aClassAliases, $bViewLink, $aDefaultLists) - { - $oSettings = new DataTableSettings($aClassAliases); - // Retrieve the class specific settings for each class/alias based on the 'list' ZList - //TODO let the caller pass some other default settings (another Zlist, extre fields...) - $aColumns = array(); - foreach ($aClassAliases as $sAlias => $sClass) - { - if ($aDefaultLists == null) - { - $aList = cmdbAbstract::FlattenZList(MetaModel::GetZListItems($sClass, 'list')); - } - else - { - $aList = $aDefaultLists[$sAlias]; - } - - $aSortOrder = MetaModel::GetOrderByDefault($sClass); - if ($bViewLink) - { - $sSort = 'none'; - if(array_key_exists('friendlyname', $aSortOrder)) - { - $sSort = $aSortOrder['friendlyname'] ? 'asc' : 'desc'; - } - $sNormalizedFName = MetaModel::NormalizeFieldSpec($sClass, 'friendlyname'); - if(array_key_exists($sNormalizedFName, $aSortOrder)) - { - $sSort = $aSortOrder[$sNormalizedFName] ? 'asc' : 'desc'; - } - - $aColumns[$sAlias]['_key_'] = $oSettings->GetFieldData($sAlias, '_key_', null, true /* bChecked */, $sSort); - } - foreach($aList as $sAttCode) - { - $sSort = 'none'; - if(array_key_exists($sAttCode, $aSortOrder)) - { - $sSort = $aSortOrder[$sAttCode] ? 'asc' : 'desc'; - } - $oAttDef = Metamodel::GetAttributeDef($sClass, $sAttCode); - $aFieldData = $oSettings->GetFieldData($sAlias, $sAttCode, $oAttDef, true /* bChecked */, $sSort); - if ($aFieldData) $aColumns[$sAlias][$sAttCode] = $aFieldData; - } - } - $iDefaultPageSize = appUserPreferences::GetPref('default_page_size', MetaModel::GetConfig()->GetMinDisplayLimit()); - $oSettings->Init($iDefaultPageSize, $aSortOrder, $aColumns); - return $oSettings; - } - - /** - * @throws \CoreException - */ - protected function FixVisibleColumns() - { - foreach($this->aClassAliases as $sAlias => $sClass) - { - if (!isset($this->aColumns[$sAlias])) - { - continue; - } - foreach($this->aColumns[$sAlias] as $sAttCode => $aData) - { - // Remove non-existent columns - // TODO: check if the existing ones are still valid (in case their type changed) - if (($sAttCode != '_key_') && (!MetaModel::IsValidAttCode($sClass, $sAttCode))) - { - unset($this->aColumns[$sAlias][$sAttCode]); - } - } - $aList = MetaModel::ListAttributeDefs($sClass); - - // Add the other (non visible ones), sorted in alphabetical order - $aTempData = array(); - foreach($aList as $sAttCode => $oAttDef) - { - if ( (!array_key_exists($sAttCode, $this->aColumns[$sAlias])) && (!($oAttDef instanceof AttributeLinkedSet || $oAttDef instanceof AttributeDashboard))) - { - $aFieldData = $this->GetFieldData($sAlias, $sAttCode, $oAttDef, false /* bChecked */, 'none'); - if ($aFieldData) $aTempData[$aFieldData['label']] = $aFieldData; - } - } - ksort($aTempData); - foreach($aTempData as $sLabel => $aFieldData) - { - $this->aColumns[$sAlias][$aFieldData['code']] = $aFieldData; - } - } - } - - /** - * @param $aClassAliases - * @param null $sTableId - * @param bool $bOnlyOnTable - * - * @return \DataTableSettings|null - * @throws \Exception - */ - static public function GetTableSettings($aClassAliases, $sTableId = null, $bOnlyOnTable = false) - { - $pref = null; - $oSettings = new DataTableSettings($aClassAliases, $sTableId); - - if ($sTableId != null) - { - // An identified table, let's fetch its own settings (if any) - $pref = appUserPreferences::GetPref($oSettings->GetPrefsKey($sTableId), null); - } - - if ($pref == null) - { - if (!$bOnlyOnTable) - { - // Try the global preferred values for this class / set of classes - $pref = appUserPreferences::GetPref($oSettings->GetPrefsKey(null), null); - } - if ($pref == null) - { - // no such settings, use the default values provided by the data model - return null; - } - } - $oSettings->unserialize($pref); - - return $oSettings; - } - - /** - * @return array - */ - public function GetSortOrder() - { - $aSortOrder = array(); - foreach($this->aColumns as $sAlias => $aColumns) - { - foreach($aColumns as $aColumn) - { - if ($aColumn['sort'] != 'none') - { - $sCode = ($aColumn['code'] == '_key_') ? 'friendlyname' : $aColumn['code']; - $aSortOrder[$sCode] = ($aColumn['sort']=='asc'); // true for ascending, false for descending - } - } - break; // TODO: For now the Set object supports only sorting on the first class of the set - } - return $aSortOrder; - } - - /** - * @param null $sTargetTableId - * - * @return bool - */ - public function Save($sTargetTableId = null) - { - $sSaveId = is_null($sTargetTableId) ? $this->sTableId : $sTargetTableId; - if ($sSaveId == null) return false; // Cannot save, the table is not identified, use SaveAsDefault instead - - $sSettings = $this->serialize(); - appUserPreferences::SetPref($this->GetPrefsKey($sSaveId), $sSettings); - return true; - } - - /** - * @return bool - */ - public function SaveAsDefault() - { - $sSettings = $this->serialize(); - appUserPreferences::SetPref($this->GetPrefsKey(null), $sSettings); - return true; - } - - - /** - * Clear the preferences for this particular table - * @param $bResetAll boolean If true,the settings for all tables of the same class(es)/alias(es) are reset - */ - public function ResetToDefault($bResetAll) - { - if (($this->sTableId == null) && (!$bResetAll)) return false; // Cannot reset, the table is not identified, use force $bResetAll instead - if ($bResetAll) - { - // Turn the key into a suitable PCRE pattern - $sKey = $this->GetPrefsKey(null); - $sPattern = str_replace(array('|'), array('\\|'), $sKey); // escape the | character - $sPattern = '#^'.str_replace(array('*'), array('.*'), $sPattern).'$#'; // Don't use slash as the delimiter since it's used in our key to delimit aliases - appUserPreferences::UnsetPref($sPattern, true); - } - else - { - appUserPreferences::UnsetPref($this->GetPrefsKey($this->sTableId), false); - } - return true; - } - - /** - * @param null $sTableId - * - * @return string - */ - protected function GetPrefsKey($sTableId = null) - { - return static::GetAppUserPreferenceKey($this->aClassAliases, $sTableId); - } - - public static function GetAppUserPreferenceKey($aClassAliases, $sTableId) - { - if ($sTableId === null) - { - $sTableId = '*'; - } - - $aKeys = array(); - foreach($aClassAliases as $sAlias => $sClass) - { - $aKeys[] = $sAlias.'-'.$sClass; - } - return implode('/', $aKeys).'|'.$sTableId; - } - - /** - * @param $sAlias - * @param $sAttCode - * @param $oAttDef - * @param $bChecked - * @param $sSort - * - * @return array|bool - * @throws \CoreException - * @throws \DictExceptionMissingString - */ - protected function GetFieldData($sAlias, $sAttCode, $oAttDef, $bChecked, $sSort) - { - $ret = false; - if ($sAttCode == '_key_') - { - $sLabel = Dict::Format('UI:ExtKey_AsLink', MetaModel::GetName($this->aClassAliases[$sAlias])); - $ret = array( - 'label' => $sLabel, - 'checked' => true, - 'disabled' => true, - 'alias' => $sAlias, - 'code' => $sAttCode, - 'sort' => $sSort, - ); - } - else if (!$oAttDef->IsLinkSet()) - { - $sLabel = $oAttDef->GetLabel(); - if ($oAttDef->IsExternalKey()) - { - $sLabel = Dict::Format('UI:ExtKey_AsLink', $oAttDef->GetLabel()); - } - else if ($oAttDef->IsExternalField()) - { - if ($oAttDef->IsFriendlyName()) - { - $sLabel = Dict::Format('UI:ExtKey_AsFriendlyName', $oAttDef->GetLabel()); - } - else - { - $oExtAttDef = $oAttDef->GetExtAttDef(); - $sLabel = Dict::Format('UI:ExtField_AsRemoteField', $oAttDef->GetLabel(), $oExtAttDef->GetLabel()); - } - } - elseif ($oAttDef instanceof AttributeFriendlyName) - { - $sLabel = Dict::Format('UI:ExtKey_AsFriendlyName', $oAttDef->GetLabel()); - } - $ret = array( - 'label' => $sLabel, - 'checked' => $bChecked, - 'disabled' => false, - 'alias' => $sAlias, - 'code' => $sAttCode, - 'sort' => $sSort, - ); - } - return $ret; - } -} diff --git a/application/displayblock.class.inc.php b/application/displayblock.class.inc.php index fbc10fb2d..8c1878ee5 100644 --- a/application/displayblock.class.inc.php +++ b/application/displayblock.class.inc.php @@ -18,13 +18,16 @@ */ use Combodo\iTop\Application\UI\Component\Alert\AlertFactory; +use Combodo\iTop\Application\Search\SearchForm; use Combodo\iTop\Application\UI\Component\Badge\BadgeFactory; use Combodo\iTop\Application\UI\Component\Button\ButtonFactory; use Combodo\iTop\Application\UI\Component\Dashlet\DashletFactory; +use Combodo\iTop\Application\UI\Component\DataTable\DataTableFactory; use Combodo\iTop\Application\UI\Component\Html\Html; use Combodo\iTop\Application\UI\Component\Toolbar\Toolbar; use Combodo\iTop\Application\UI\iUIBlock; use Combodo\iTop\Application\UI\Layout\UIContentBlock; +use Combodo\iTop\Renderer\BlockRenderer; require_once(APPROOT.'/application/utils.inc.php'); @@ -43,6 +46,7 @@ class DisplayBlock protected $m_sStyle; protected $m_bAsynchronous; protected $m_aParams; + /** @var \DBObjectSet|null */ protected $m_oSet; protected $m_bShowObsoleteData = null; @@ -239,7 +243,6 @@ class DisplayBlock $sFilter = addslashes($this->m_oFilter->serialize(false, $aQueryParams)); // Used either for asynchronous or auto_reload if (!$this->m_bAsynchronous) { // render now - $oHtml->AddHtml("
\n"); try { $oHtml->AddSubBlock($this->GetRenderContent($oPage, $aExtraParams, $sId)); } catch (Exception $e) { @@ -253,12 +256,10 @@ HTML; } IssueLog::Error('Exception during GetDisplay: '.$e->getMessage()); } - $oHtml->AddHtml("
\n"); } else { // render it as an Ajax (asynchronous) call - $oHtml->AddHtml("
\n"); + $oHtml->AddCSSClasses("display_block loading"); $oHtml->AddHtml($oPage->GetP(" ".Dict::S('UI:Loading'))); - $oHtml->AddHtml("
\n"); $oPage->add_script(' $.post("ajax.render.php?style='.$this->m_sStyle.'", { operation: "ajax", filter: "'.$sFilter.'", extra_params: "'.$sExtraParams.'" }, @@ -426,75 +427,8 @@ HTML; switch($this->m_sStyle) { case 'count': - if (isset($aExtraParams['group_by'])) - { - $this->MakeGroupByQuery($aExtraParams, $oGroupByExp, $sGroupByLabel, $aGroupBy, $sAggregationFunction, $sFctVar, $sAggregationAttr, $sSql); - - $aRes = CMDBSource::QueryToArray($sSql); - - $aGroupBy = array(); - $aLabels = array(); - $aValues = array(); - $iTotalCount = 0; - foreach ($aRes as $iRow => $aRow) - { - $sValue = $aRow['grouped_by_1']; - $aValues[$iRow] = $sValue; - $sHtmlValue = $oGroupByExp->MakeValueLabel($this->m_oFilter, $sValue, $sValue); - $aLabels[$iRow] = $sHtmlValue; - $aGroupBy[$iRow] = (int) $aRow[$sFctVar]; - $iTotalCount += $aRow['_itop_count_']; - } - - - $aData = array(); - $oAppContext = new ApplicationContext(); - $sParams = $oAppContext->GetForLink(); - foreach($aGroupBy as $iRow => $iCount) - { - // Build the search for this subset - $oSubsetSearch = $this->m_oFilter->DeepClone(); - $oCondition = new BinaryExpression($oGroupByExp, '=', new ScalarExpression($aValues[$iRow])); - $oSubsetSearch->AddConditionExpression($oCondition); - if (isset($aExtraParams['query_params'])) - { - $aQueryParams = $aExtraParams['query_params']; - } - else - { - $aQueryParams = array(); - } - $sFilter = rawurlencode($oSubsetSearch->serialize(false, $aQueryParams)); - - $aData[] = array ('group' => $aLabels[$iRow], - 'value' => "$iCount"); // TO DO: add the context information - } - $aAttribs =array( - 'group' => array('label' => $sGroupByLabel, 'description' => ''), - 'value' => array( - 'label' => Dict::S('UI:GroupBy:'.$sAggregationFunction), - 'description' => Dict::Format('UI:GroupBy:'.$sAggregationFunction.'+', $sAggregationAttr), - ), - ); - $sFormat = isset($aExtraParams['format']) ? $aExtraParams['format'] : 'UI:Pagination:HeaderNoSelection'; - $sHtml .= $oPage->GetP(Dict::Format($sFormat, $iTotalCount)); - $sHtml .= $oPage->GetTable($aAttribs, $aData); - - $oPage->add_ready_script("LoadGroupBySortOrder('$sId');\n$('#{$sId} table.listResults').unbind('sortEnd.group_by').bind('sortEnd.group_by', function() { SaveGroupBySortOrder('$sId', $(this)[0].config.sortList); })"); - } - else - { - // Simply count the number of elements in the set - $iCount = $this->m_oSet->Count(); - $sFormat = 'UI:CountOfObjects'; - if (isset($aExtraParams['format'])) - { - $sFormat = $aExtraParams['format']; - } - $sHtml .= $oPage->GetP(Dict::Format($sFormat, $iCount)); - } - - break; + $oBlock = $this->RenderCount($aExtraParams, $oPage, $sId); + break; case 'join': $aDisplayAliases = isset($aExtraParams['display_aliases']) ? explode(',', $aExtraParams['display_aliases']): array(); @@ -582,7 +516,16 @@ HTML; $index++; } $oSet = new CMDBObjectSet($oBlockFilter, array(), $aArgs); - $sHtml .= "".cmdbAbstractObject::GetDisplayExtendedSet($oPage, $oSet, $aExtraParams)."\n"; + if (empty($aExtraParams['currentId'])) + { + $iListId = $oPage->GetUniqueId(); // Works only if not in an Ajax page !! + } + else + { + $iListId = $aExtraParams['currentId']; + } + $oBlock = DataTableFactory::MakeForRendering( $iListId, $this->m_oSet, $aExtraParams); + $sHtml .= "".BlockRenderer::RenderBlockTemplates($oBlock)."\n"; } } $sHtml .= "\n"; @@ -591,96 +534,11 @@ HTML; break; case 'list_search': + $oBlock = $this->RenderListSearch($aExtraParams, $oPage); + break; + case 'list': - $aClasses = $this->m_oSet->GetSelectedClasses(); - $aAuthorizedClasses = array(); - if (count($aClasses) > 1) - { - // Check the classes that can be read (i.e authorized) by this user... - foreach($aClasses as $sAlias => $sClassName) - { - if (UserRights::IsActionAllowed($sClassName, UR_ACTION_READ, $this->m_oSet) != UR_ALLOWED_NO) - { - $aAuthorizedClasses[$sAlias] = $sClassName; - } - } - if (count($aAuthorizedClasses) > 0) - { - if($this->m_oSet->CountWithLimit(1) > 0) - { - $sHtml .= cmdbAbstractObject::GetDisplayExtendedSet($oPage, $this->m_oSet, $aExtraParams); - } - else - { - // Empty set - $sHtml .= $oPage->GetP(Dict::S('UI:NoObjectToDisplay')); - } - } - else - { - // Not authorized - $sHtml .= $oPage->GetP(Dict::S('UI:NoObjectToDisplay')); - } - } - else - { - // The list is made of only 1 class of objects, actions on the list are possible - if ( ($this->m_oSet->CountWithLimit(1)> 0) && (UserRights::IsActionAllowed($this->m_oSet->GetClass(), UR_ACTION_READ, $this->m_oSet) == UR_ALLOWED_YES) ) - { - $sHtml .= cmdbAbstractObject::GetDisplaySet($oPage, $this->m_oSet, $aExtraParams); - } - else - { - $sHtml .= $oPage->GetP(Dict::S('UI:NoObjectToDisplay')); - $sClass = $this->m_oFilter->GetClass(); - $bDisplayMenu = isset($aExtraParams['menu']) ? $aExtraParams['menu'] == true : true; - if ($bDisplayMenu) - { - if ((UserRights::IsActionAllowed($sClass, UR_ACTION_MODIFY) == UR_ALLOWED_YES)) - { - $sLinkTarget = ''; - $oAppContext = new ApplicationContext(); - $sParams = $oAppContext->GetForLink(); - // 1:n links, populate the target object as a default value when creating a new linked object - if (isset($aExtraParams['target_attr'])) - { - $sLinkTarget = ' target="_blank" '; - $aExtraParams['default'][$aExtraParams['target_attr']] = $aExtraParams['object_id']; - } - $sDefault = ''; - if (!empty($aExtraParams['default'])) - { - foreach($aExtraParams['default'] as $sKey => $sValue) - { - $sDefault.= "&default[$sKey]=$sValue"; - } - } - - $sHtml .= $oPage->GetP("".Dict::Format('UI:ClickToCreateNew', Metamodel::GetName($sClass))."\n"); - } - } - } - - if (isset($aExtraParams['update_history']) && true == $aExtraParams['update_history']) - { - $sSearchFilter = $this->m_oSet->GetFilter()->serialize(); - // Limit the size of the URL (N°1585 - request uri too long) - if (strlen($sSearchFilter) < SERVER_MAX_URL_LENGTH) - { - $seventAttachedData = json_encode(array( - 'filter' => $sSearchFilter, - 'breadcrumb_id' => "ui-search-".$this->m_oSet->GetClass(), - 'breadcrumb_label' => MetaModel::GetName($this->m_oSet->GetClass()), - 'breadcrumb_max_count' => utils::GetConfig()->Get('breadcrumb.max_count'), - 'breadcrumb_instance_id' => MetaModel::GetConfig()->GetItopInstanceid(), - 'breadcrumb_icon' => 'fas fa-search', - 'breadcrumb_icon_type' => iTopWebPage::ENUM_BREADCRUMB_ENTRY_ICON_TYPE_CSS_CLASSES, - )); - - $oPage->add_ready_script("$('body').trigger('update_history.itop', [$seventAttachedData])"); - } - } - } + $oBlock = $this->RenderList($aExtraParams, $oPage); break; case 'links': @@ -791,14 +649,8 @@ HTML; break; case 'search': - if (!$oPage->IsPrintableVersion()) - { - $sHtml .= "
\n"; - $aExtraParams['currentId'] = $sId; - $sHtml .= cmdbAbstractObject::GetSearchForm($oPage, $this->m_oSet, $aExtraParams); - $sHtml .= "
\n"; - } - break; + $oBlock = $this->RenderSearch($oPage, $sId, $aExtraParams); + break; case 'chart': static $iChartCounter = 0; @@ -1377,6 +1229,206 @@ JS return $oBlock; } + + /** + * @param array $aExtraParams + * @param \WebPage $oPage + * @param string|null $sId + * + * @return \Combodo\iTop\Application\UI\iUIBlock + * @throws \ArchivedObjectException + * @throws \CoreException + * @throws \MissingQueryArgument + * @throws \MySQLException + * @throws \MySQLHasGoneAwayException + * @throws \Exception + */ + protected function RenderCount(array $aExtraParams, WebPage $oPage, ?string $sId): iUIBlock + { + if (isset($aExtraParams['group_by'])) { + $this->MakeGroupByQuery($aExtraParams, $oGroupByExp, $sGroupByLabel, $aGroupBy, $sAggregationFunction, $sFctVar, $sAggregationAttr, $sSql); + + $aRes = CMDBSource::QueryToArray($sSql); + + $aGroupBy = array(); + $aLabels = array(); + $aValues = array(); + $iTotalCount = 0; + foreach ($aRes as $iRow => $aRow) { + $sValue = $aRow['grouped_by_1']; + $aValues[$iRow] = $sValue; + $sHtmlValue = $oGroupByExp->MakeValueLabel($this->m_oFilter, $sValue, $sValue); + $aLabels[$iRow] = $sHtmlValue; + $aGroupBy[$iRow] = (int)$aRow[$sFctVar]; + $iTotalCount += $aRow['_itop_count_']; + } + + $aData = array(); + $oAppContext = new ApplicationContext(); + $sParams = $oAppContext->GetForLink(); + foreach ($aGroupBy as $iRow => $iCount) { + // Build the search for this subset + $oSubsetSearch = $this->m_oFilter->DeepClone(); + $oCondition = new BinaryExpression($oGroupByExp, '=', new ScalarExpression($aValues[$iRow])); + $oSubsetSearch->AddConditionExpression($oCondition); + if (isset($aExtraParams['query_params'])) { + $aQueryParams = $aExtraParams['query_params']; + } else { + $aQueryParams = array(); + } + $sFilter = rawurlencode($oSubsetSearch->serialize(false, $aQueryParams)); + + $aData[] = array( + 'group' => $aLabels[$iRow], + 'value' => "$iCount" + ); // TO DO: add the context information + } + $aAttribs = array( + 'group' => array('label' => $sGroupByLabel, 'description' => ''), + 'value' => array( + 'label' => Dict::S('UI:GroupBy:'.$sAggregationFunction), + 'description' => Dict::Format('UI:GroupBy:'.$sAggregationFunction.'+', $sAggregationAttr), + ), + ); + $sFormat = isset($aExtraParams['format']) ? $aExtraParams['format'] : 'UI:Pagination:HeaderNoSelection'; + $sTitle = Dict::Format($sFormat, $iTotalCount); + $oBlock = DataTableFactory::MakeForStaticData($sTitle, $aAttribs, $aData); + + // $oPage->add_ready_script("LoadGroupBySortOrder('$sId');\n$('#{$sId} table.listResults').unbind('sortEnd.group_by').bind('sortEnd.group_by', function() { SaveGroupBySortOrder('$sId', $(this)[0].config.sortList); })"); + } else { + // Simply count the number of elements in the set + $iCount = $this->m_oSet->Count(); + $sFormat = 'UI:CountOfObjects'; + if (isset($aExtraParams['format'])) { + $sFormat = $aExtraParams['format']; + } + $oBlock = new Html($oPage->GetP(Dict::Format($sFormat, $iCount))); + } + return $oBlock; +} + + /** + * @param \WebPage $oPage + * @param string|null $sId + * @param array $aExtraParams + * + * @return \Combodo\iTop\Application\UI\iUIBlock + */ + protected function RenderSearch(WebPage $oPage, ?string $sId, array $aExtraParams): iUIBlock + { + $oBlock = null; + + if (!$oPage->IsPrintableVersion()) { + $aExtraParams['currentId'] = $sId; + $oSearchForm = new SearchForm(); + $oBlock = $oSearchForm->GetSearchFormUIBlock($oPage, $this->m_oSet, $aExtraParams); + } + + return $oBlock; + } + + protected function RenderListSearch(array $aExtraParams, WebPage $oPage) + { + return $this->RenderList($aExtraParams, $oPage); + } + + /** + * @param array $aExtraParams + * @param \WebPage $oPage + * + * @throws \ArchivedObjectException + * @throws \ConfigException + * @throws \CoreException + * @throws \CoreUnexpectedValue + * @throws \DictExceptionMissingString + * @throws \MissingQueryArgument + * @throws \MySQLException + * @throws \MySQLHasGoneAwayException + * @throws \OQLException + * @throws \ReflectionException + */ + protected function RenderList(array $aExtraParams, WebPage $oPage) + { + $aClasses = $this->m_oSet->GetSelectedClasses(); + $aAuthorizedClasses = array(); + $oBlock = null; + $sHtml = ''; + if (count($aClasses) > 1) { + // Check the classes that can be read (i.e authorized) by this user... + foreach ($aClasses as $sAlias => $sClassName) { + if (UserRights::IsActionAllowed($sClassName, UR_ACTION_READ, $this->m_oSet) != UR_ALLOWED_NO) { + $aAuthorizedClasses[$sAlias] = $sClassName; + } + } + if (count($aAuthorizedClasses) > 0) { + if ($this->m_oSet->CountWithLimit(1) > 0) { + if (empty($aExtraParams['currentId'])) { + $iListId = $oPage->GetUniqueId(); // Works only if not in an Ajax page !! + } else { + $iListId = $aExtraParams['currentId']; + } + $oBlock = DataTableFactory::MakeForObject($oPage, $iListId, $this->m_oSet, $aExtraParams); + } else { + // Empty set + $sHtml .= $oPage->GetP(Dict::S('UI:NoObjectToDisplay')); + } + } else { + // Not authorized + $sHtml .= $oPage->GetP(Dict::S('UI:NoObjectToDisplay')); + } + } else { + // The list is made of only 1 class of objects, actions on the list are possible + if (($this->m_oSet->CountWithLimit(1) > 0) && (UserRights::IsActionAllowed($this->m_oSet->GetClass(), UR_ACTION_READ, $this->m_oSet) == UR_ALLOWED_YES)) { + $oBlock = cmdbAbstractObject::GetDisplaySetBlock($oPage, $this->m_oSet, $aExtraParams); + } else { + $sHtml .= $oPage->GetP(Dict::S('UI:NoObjectToDisplay')); + $sClass = $this->m_oFilter->GetClass(); + $bDisplayMenu = isset($aExtraParams['menu']) ? $aExtraParams['menu'] == true : true; + if ($bDisplayMenu) { + if ((UserRights::IsActionAllowed($sClass, UR_ACTION_MODIFY) == UR_ALLOWED_YES)) { + $sLinkTarget = ''; + $oAppContext = new ApplicationContext(); + $sParams = $oAppContext->GetForLink(); + // 1:n links, populate the target object as a default value when creating a new linked object + if (isset($aExtraParams['target_attr'])) { + $sLinkTarget = ' target="_blank" '; + $aExtraParams['default'][$aExtraParams['target_attr']] = $aExtraParams['object_id']; + } + $sDefault = ''; + if (!empty($aExtraParams['default'])) { + foreach ($aExtraParams['default'] as $sKey => $sValue) { + $sDefault .= "&default[$sKey]=$sValue"; + } + } + + $sHtml .= $oPage->GetP("".Dict::Format('UI:ClickToCreateNew', Metamodel::GetName($sClass))."\n"); + } + } + } + + if (isset($aExtraParams['update_history']) && true == $aExtraParams['update_history']) { + $sSearchFilter = $this->m_oSet->GetFilter()->serialize(); + // Limit the size of the URL (N°1585 - request uri too long) + if (strlen($sSearchFilter) < SERVER_MAX_URL_LENGTH) { + $seventAttachedData = json_encode(array( + 'filter' => $sSearchFilter, + 'breadcrumb_id' => "ui-search-".$this->m_oSet->GetClass(), + 'breadcrumb_label' => MetaModel::GetName($this->m_oSet->GetClass()), + 'breadcrumb_max_count' => utils::GetConfig()->Get('breadcrumb.max_count'), + 'breadcrumb_instance_id' => MetaModel::GetConfig()->GetItopInstanceid(), + 'breadcrumb_icon' => 'fas fa-search', + 'breadcrumb_icon_type' => iTopWebPage::ENUM_BREADCRUMB_ENTRY_ICON_TYPE_CSS_CLASSES, + )); + + $oPage->add_ready_script("$('body').trigger('update_history.itop', [$seventAttachedData])"); + } + } + } + if (is_null($oBlock)) { + $oBlock = new Html($sHtml); + } + return $oBlock; + } } /** @@ -1552,244 +1604,308 @@ class MenuBlock extends DisplayBlock { $this->m_sStyle = 'list'; } - $oAppContext = new ApplicationContext(); - $sContext = $oAppContext->GetForLink(); - if (!empty($sContext)) { - $sContext = '&'.$sContext; - } $sClass = $this->m_oFilter->GetClass(); - $oReflectionClass = new ReflectionClass($sClass); $oSet = new CMDBObjectSet($this->m_oFilter); - $sFilter = $this->m_oFilter->serialize(); - $aActions = []; - $sUIPage = cmdbAbstractObject::ComputeStandardUIPage($sClass); - $sRootUrl = utils::GetAbsoluteUrlAppRoot(); - // Common params that will be applied to actions - $aActionParams = array(); - if (isset($aExtraParams['menu_actions_target'])) { - $aActionParams['target'] = $aExtraParams['menu_actions_target']; - } - // 1:n links, populate the target object as a default value when creating a new linked object - if (isset($aExtraParams['target_attr'])) { - $aExtraParams['default'][$aExtraParams['target_attr']] = $aExtraParams['object_id']; - } - $sDefault = ''; - if (!empty($aExtraParams['default'])) { - foreach ($aExtraParams['default'] as $sKey => $sValue) { - $sDefault .= "&default[$sKey]=$sValue"; - } - } - $bIsCreationAllowed = (UserRights::IsActionAllowed($sClass, UR_ACTION_CREATE) == UR_ALLOWED_YES) && ($oReflectionClass->IsSubclassOf('cmdbAbstractObject')); $sRefreshAction = ''; - switch ($oSet->Count()) { - case 0: - // No object in the set, the only possible action is "new" - if ($bIsCreationAllowed) { - $aActions['UI:Menu:New'] = array('label' => Dict::S('UI:Menu:New'), 'url' => "{$sRootUrl}pages/$sUIPage?operation=new&class=$sClass{$sContext}{$sDefault}") + $aActionParams; + if (!isset($aExtraParams['selection_mode']) || $aExtraParams['selection_mode'] == "") { + $oAppContext = new ApplicationContext(); + $sContext = $oAppContext->GetForLink(); + if (!empty($sContext)) { + $sContext = '&'.$sContext; + } + $oReflectionClass = new ReflectionClass($sClass); + $sFilter = $this->m_oFilter->serialize(); + $aActions = []; + $sUIPage = cmdbAbstractObject::ComputeStandardUIPage($sClass); + $sRootUrl = utils::GetAbsoluteUrlAppRoot(); + // Common params that will be applied to actions + $aActionParams = array(); + if (isset($aExtraParams['menu_actions_target'])) { + $aActionParams['target'] = $aExtraParams['menu_actions_target']; + } + // 1:n links, populate the target object as a default value when creating a new linked object + if (isset($aExtraParams['target_attr'])) { + $aExtraParams['default'][$aExtraParams['target_attr']] = $aExtraParams['object_id']; + } + $sDefault = ''; + if (!empty($aExtraParams['default'])) { + foreach ($aExtraParams['default'] as $sKey => $sValue) { + $sDefault .= "&default[$sKey]=$sValue"; } - break; + } + $bIsCreationAllowed = (UserRights::IsActionAllowed($sClass, + UR_ACTION_CREATE) == UR_ALLOWED_YES) && ($oReflectionClass->IsSubclassOf('cmdbAbstractObject')); + switch ($oSet->Count()) { + case 0: + // No object in the set, the only possible action is "new" + if ($bIsCreationAllowed) { + $aActions['UI:Menu:New'] = array( + 'label' => Dict::S('UI:Menu:New'), + 'url' => "{$sRootUrl}pages/$sUIPage?operation=new&class=$sClass{$sContext}{$sDefault}" + ) + $aActionParams; + } + break; - case 1: - $oObj = $oSet->Fetch(); - if (is_null($oObj)) { - if (!isset($aExtraParams['link_attr'])) { - if ($bIsCreationAllowed) { - $aActions['UI:Menu:New'] = array('label' => Dict::S('UI:Menu:New'), 'url' => "{$sRootUrl}pages/$sUIPage?operation=new&class=$sClass{$sContext}{$sDefault}") + $aActionParams; + case 1: + $oObj = $oSet->Fetch(); + if (is_null($oObj)) { + if (!isset($aExtraParams['link_attr'])) { + if ($bIsCreationAllowed) { + $aActions['UI:Menu:New'] = array( + 'label' => Dict::S('UI:Menu:New'), + 'url' => "{$sRootUrl}pages/$sUIPage?operation=new&class=$sClass{$sContext}{$sDefault}" + ) + $aActionParams; + } } - } - } else { - $id = $oObj->GetKey(); - if (utils::ReadParam('operation') == 'details') { - if ($_SERVER['REQUEST_METHOD'] == 'GET') { - $sRefreshAction = "window.location.reload();"; - } else { - $sRefreshAction = "window.location.href='".ApplicationContext::MakeObjectUrl(get_class($oObj), $id)."';"; + } else { + $id = $oObj->GetKey(); + if (utils::ReadParam('operation') == 'details') { + if ($_SERVER['REQUEST_METHOD'] == 'GET') { + $sRefreshAction = "window.location.reload();"; + } else { + $sRefreshAction = "window.location.href='".ApplicationContext::MakeObjectUrl(get_class($oObj), $id)."';"; + } } - } - $bLocked = false; - if (MetaModel::GetConfig()->Get('concurrent_lock_enabled')) { - $aLockInfo = iTopOwnershipLock::IsLocked(get_class($oObj), $id); - if ($aLockInfo['locked']) { - $bLocked = true; - //$this->AddMenuSeparator($aActions); - //$aActions['concurrent_lock_unlock'] = array ('label' => Dict::S('UI:Menu:ReleaseConcurrentLock'), 'url' => "{$sRootUrl}pages/$sUIPage?operation=kill_lock&class=$sClass&id=$id{$sContext}"); + $bLocked = false; + if (MetaModel::GetConfig()->Get('concurrent_lock_enabled')) { + $aLockInfo = iTopOwnershipLock::IsLocked(get_class($oObj), $id); + if ($aLockInfo['locked']) { + $bLocked = true; + //$this->AddMenuSeparator($aActions); + //$aActions['concurrent_lock_unlock'] = array ('label' => Dict::S('UI:Menu:ReleaseConcurrentLock'), 'url' => "{$sRootUrl}pages/$sUIPage?operation=kill_lock&class=$sClass&id=$id{$sContext}"); + } } - } - $bRawModifiedAllowed = (UserRights::IsActionAllowed($sClass, UR_ACTION_MODIFY, $oSet) == UR_ALLOWED_YES) && ($oReflectionClass->IsSubclassOf('cmdbAbstractObject')); - $bIsModifyAllowed = !$bLocked && $bRawModifiedAllowed; - $bIsDeleteAllowed = !$bLocked && UserRights::IsActionAllowed($sClass, UR_ACTION_DELETE, $oSet); - // Just one object in the set, possible actions are "new / clone / modify and delete" - if (!isset($aExtraParams['link_attr'])) { - if ($bIsModifyAllowed) { - $aActions['UI:Menu:Modify'] = array('label' => Dict::S('UI:Menu:Modify'), 'url' => "{$sRootUrl}pages/$sUIPage?operation=modify&class=$sClass&id=$id{$sContext}#") + $aActionParams; - } - if ($bIsCreationAllowed) { - $aActions['UI:Menu:New'] = array('label' => Dict::S('UI:Menu:New'), 'url' => "{$sRootUrl}pages/$sUIPage?operation=new&class=$sClass{$sContext}{$sDefault}") + $aActionParams; - } - if ($bIsDeleteAllowed) { - $aActions['UI:Menu:Delete'] = array('label' => Dict::S('UI:Menu:Delete'), 'url' => "{$sRootUrl}pages/$sUIPage?operation=delete&class=$sClass&id=$id{$sContext}") + $aActionParams; - } - // Transitions / Stimuli - if (!$bLocked) { - $aTransitions = $oObj->EnumTransitions(); - if (count($aTransitions)) { + $bRawModifiedAllowed = (UserRights::IsActionAllowed($sClass, UR_ACTION_MODIFY, + $oSet) == UR_ALLOWED_YES) && ($oReflectionClass->IsSubclassOf('cmdbAbstractObject')); + $bIsModifyAllowed = !$bLocked && $bRawModifiedAllowed; + $bIsDeleteAllowed = !$bLocked && UserRights::IsActionAllowed($sClass, UR_ACTION_DELETE, $oSet); + // Just one object in the set, possible actions are "new / clone / modify and delete" + if (!isset($aExtraParams['link_attr'])) { + if ($bIsModifyAllowed) { + $aActions['UI:Menu:Modify'] = array( + 'label' => Dict::S('UI:Menu:Modify'), + 'url' => "{$sRootUrl}pages/$sUIPage?operation=modify&class=$sClass&id=$id{$sContext}#" + ) + $aActionParams; + } + if ($bIsCreationAllowed) { + $aActions['UI:Menu:New'] = array( + 'label' => Dict::S('UI:Menu:New'), + 'url' => "{$sRootUrl}pages/$sUIPage?operation=new&class=$sClass{$sContext}{$sDefault}" + ) + $aActionParams; + } + if ($bIsDeleteAllowed) { + $aActions['UI:Menu:Delete'] = array( + 'label' => Dict::S('UI:Menu:Delete'), + 'url' => "{$sRootUrl}pages/$sUIPage?operation=delete&class=$sClass&id=$id{$sContext}" + ) + $aActionParams; + } + // Transitions / Stimuli + if (!$bLocked) { + $aTransitions = $oObj->EnumTransitions(); + if (count($aTransitions)) { + $this->AddMenuSeparator($aActions); + $aStimuli = Metamodel::EnumStimuli(get_class($oObj)); + foreach ($aTransitions as $sStimulusCode => $aTransitionDef) { + $iActionAllowed = (get_class($aStimuli[$sStimulusCode]) == 'StimulusUserAction') ? UserRights::IsStimulusAllowed($sClass, + $sStimulusCode, $oSet) : UR_ALLOWED_NO; + switch ($iActionAllowed) { + case UR_ALLOWED_YES: + $aActions[$sStimulusCode] = array( + 'label' => $aStimuli[$sStimulusCode]->GetLabel(), + 'url' => "{$sRootUrl}pages/UI.php?operation=stimulus&stimulus=$sStimulusCode&class=$sClass&id=$id{$sContext}" + ) + $aActionParams; + break; + + default: + // Do nothing + } + } + } + } + // Relations... + $aRelations = MetaModel::EnumRelationsEx($sClass); + if (count($aRelations)) { $this->AddMenuSeparator($aActions); - $aStimuli = Metamodel::EnumStimuli(get_class($oObj)); - foreach ($aTransitions as $sStimulusCode => $aTransitionDef) { - $iActionAllowed = (get_class($aStimuli[$sStimulusCode]) == 'StimulusUserAction') ? UserRights::IsStimulusAllowed($sClass, $sStimulusCode, $oSet) : UR_ALLOWED_NO; - switch ($iActionAllowed) { - case UR_ALLOWED_YES: - $aActions[$sStimulusCode] = array('label' => $aStimuli[$sStimulusCode]->GetLabel(), 'url' => "{$sRootUrl}pages/UI.php?operation=stimulus&stimulus=$sStimulusCode&class=$sClass&id=$id{$sContext}") + $aActionParams; - break; + foreach ($aRelations as $sRelationCode => $aRelationInfo) { + if (array_key_exists('down', $aRelationInfo)) { + $aActions[$sRelationCode.'_down'] = array( + 'label' => $aRelationInfo['down'], + 'url' => "{$sRootUrl}pages/$sUIPage?operation=view_relations&relation=$sRelationCode&direction=down&class=$sClass&id=$id{$sContext}" + ) + $aActionParams; + } + if (array_key_exists('up', $aRelationInfo)) { + $aActions[$sRelationCode.'_up'] = array( + 'label' => $aRelationInfo['up'], + 'url' => "{$sRootUrl}pages/$sUIPage?operation=view_relations&relation=$sRelationCode&direction=up&class=$sClass&id=$id{$sContext}" + ) + $aActionParams; + } + } + } + if ($bLocked && $bRawModifiedAllowed) { + // Add a special menu to kill the lock, but only to allowed users who can also modify this object + /** @var array $aAllowedProfiles */ + $aAllowedProfiles = MetaModel::GetConfig()->Get('concurrent_lock_override_profiles'); + $bCanKill = false; - default: - // Do nothing + $oUser = UserRights::GetUserObject(); + $aUserProfiles = array(); + if (!is_null($oUser)) { + $oProfileSet = $oUser->Get('profile_list'); + while ($oProfile = $oProfileSet->Fetch()) { + $aUserProfiles[$oProfile->Get('profile')] = true; + } + } + + foreach ($aAllowedProfiles as $sProfile) { + if (array_key_exists($sProfile, $aUserProfiles)) { + $bCanKill = true; + break; + } + } + + if ($bCanKill) { + $this->AddMenuSeparator($aActions); + $aActions['concurrent_lock_unlock'] = array( + 'label' => Dict::S('UI:Menu:KillConcurrentLock'), + 'url' => "{$sRootUrl}pages/$sUIPage?operation=kill_lock&class=$sClass&id=$id{$sContext}" + ); + } + } + } + $this->AddMenuSeparator($aActions); + /** @var \iApplicationUIExtension $oExtensionInstance */ + foreach (MetaModel::EnumPlugins('iApplicationUIExtension') as $oExtensionInstance) { + $oSet->Rewind(); + foreach ($oExtensionInstance->EnumAllowedActions($oSet) as $sLabel => $sUrl) { + $aActions[$sLabel] = array('label' => $sLabel, 'url' => $sUrl) + $aActionParams; + } + } + } + break; + + default: + // Check rights + // New / Modify + $bIsModifyAllowed = UserRights::IsActionAllowed($sClass, UR_ACTION_MODIFY, + $oSet) && ($oReflectionClass->IsSubclassOf('cmdbAbstractObject')); + $bIsBulkModifyAllowed = (!MetaModel::IsAbstract($sClass)) && UserRights::IsActionAllowed($sClass, UR_ACTION_BULK_MODIFY, + $oSet) && ($oReflectionClass->IsSubclassOf('cmdbAbstractObject')); + $bIsBulkDeleteAllowed = UserRights::IsActionAllowed($sClass, UR_ACTION_BULK_DELETE, $oSet); + if (isset($aExtraParams['link_attr'])) { + $id = $aExtraParams['object_id']; + $sTargetAttr = $aExtraParams['target_attr']; + $oAttDef = MetaModel::GetAttributeDef($sClass, $sTargetAttr); + $sTargetClass = $oAttDef->GetTargetClass(); + if ($bIsModifyAllowed) { + $aActions['UI:Menu:Add'] = array( + 'label' => Dict::S('UI:Menu:Add'), + 'url' => "{$sRootUrl}pages/$sUIPage?operation=modify_links&class=$sClass&link_attr=".$aExtraParams['link_attr']."&target_class=$sTargetClass&id=$id&addObjects=true{$sContext}" + ) + $aActionParams; + } + if ($bIsBulkModifyAllowed) { + $aActions['UI:Menu:Manage'] = array( + 'label' => Dict::S('UI:Menu:Manage'), + 'url' => "{$sRootUrl}pages/$sUIPage?operation=modify_links&class=$sClass&link_attr=".$aExtraParams['link_attr']."&target_class=$sTargetClass&id=$id{$sContext}" + ) + $aActionParams; + } + //if ($bIsBulkDeleteAllowed) { $aActions[] = array ('label' => 'Remove All...', 'url' => "#") + $aActionParams; } + } else { + // many objects in the set, possible actions are: new / modify all / delete all + if ($bIsCreationAllowed) { + $aActions['UI:Menu:New'] = array( + 'label' => Dict::S('UI:Menu:New'), + 'url' => "{$sRootUrl}pages/$sUIPage?operation=new&class=$sClass{$sContext}{$sDefault}" + ) + $aActionParams; + } + if ($bIsBulkModifyAllowed) { + $aActions['UI:Menu:ModifyAll'] = array( + 'label' => Dict::S('UI:Menu:ModifyAll'), + 'url' => "{$sRootUrl}pages/$sUIPage?operation=select_for_modify_all&class=$sClass&filter=".urlencode($sFilter)."{$sContext}" + ) + $aActionParams; + } + if ($bIsBulkDeleteAllowed) { + $aActions['UI:Menu:BulkDelete'] = array( + 'label' => Dict::S('UI:Menu:BulkDelete'), + 'url' => "{$sRootUrl}pages/$sUIPage?operation=select_for_deletion&filter=".urlencode($sFilter)."{$sContext}" + ) + $aActionParams; + } + + // Stimuli + $aStates = MetaModel::EnumStates($sClass); + // Do not perform time consuming computations if there are too may objects in the list + $iLimit = MetaModel::GetConfig()->Get('complex_actions_limit'); + + if ((count($aStates) > 0) && (($iLimit == 0) || ($oSet->CountWithLimit($iLimit + 1) < $iLimit))) { + // Life cycle actions may be available... if all objects are in the same state + // + // Group by + $oGroupByExp = new FieldExpression(MetaModel::GetStateAttributeCode($sClass), $this->m_oFilter->GetClassAlias()); + $aGroupBy = array('__state__' => $oGroupByExp); + $aQueryParams = array(); + if (isset($aExtraParams['query_params'])) { + $aQueryParams = $aExtraParams['query_params']; + } + $sSql = $this->m_oFilter->MakeGroupByQuery($aQueryParams, $aGroupBy); + $aRes = CMDBSource::QueryToArray($sSql); + if (count($aRes) == 1) { + // All objects are in the same state... + $sState = $aRes[0]['__state__']; + $aTransitions = Metamodel::EnumTransitions($sClass, $sState); + if (count($aTransitions)) { + $this->AddMenuSeparator($aActions); + $aStimuli = Metamodel::EnumStimuli($sClass); + foreach ($aTransitions as $sStimulusCode => $aTransitionDef) { + $oSet->Rewind(); + // As soon as the user rights implementation will browse the object set, + // then we might consider using OptimizeColumnLoad() here + $iActionAllowed = UserRights::IsStimulusAllowed($sClass, $sStimulusCode, $oSet); + $iActionAllowed = (get_class($aStimuli[$sStimulusCode]) == 'StimulusUserAction') ? $iActionAllowed : UR_ALLOWED_NO; + switch ($iActionAllowed) { + case UR_ALLOWED_YES: + case UR_ALLOWED_DEPENDS: + $aActions[$sStimulusCode] = array( + 'label' => $aStimuli[$sStimulusCode]->GetLabel(), + 'url' => "{$sRootUrl}pages/UI.php?operation=select_bulk_stimulus&stimulus=$sStimulusCode&state=$sState&class=$sClass&filter=".urlencode($sFilter)."{$sContext}" + ) + $aActionParams; + break; + + default: + // Do nothing + } } } } } - // Relations... - $aRelations = MetaModel::EnumRelationsEx($sClass); - if (count($aRelations)) { - $this->AddMenuSeparator($aActions); - foreach ($aRelations as $sRelationCode => $aRelationInfo) { - if (array_key_exists('down', $aRelationInfo)) { - $aActions[$sRelationCode.'_down'] = array('label' => $aRelationInfo['down'], 'url' => "{$sRootUrl}pages/$sUIPage?operation=view_relations&relation=$sRelationCode&direction=down&class=$sClass&id=$id{$sContext}") + $aActionParams; - } - if (array_key_exists('up', $aRelationInfo)) { - $aActions[$sRelationCode.'_up'] = array('label' => $aRelationInfo['up'], 'url' => "{$sRootUrl}pages/$sUIPage?operation=view_relations&relation=$sRelationCode&direction=up&class=$sClass&id=$id{$sContext}") + $aActionParams; - } - } - } - if ($bLocked && $bRawModifiedAllowed) { - // Add a special menu to kill the lock, but only to allowed users who can also modify this object - /** @var array $aAllowedProfiles */ - $aAllowedProfiles = MetaModel::GetConfig()->Get('concurrent_lock_override_profiles'); - $bCanKill = false; - - $oUser = UserRights::GetUserObject(); - $aUserProfiles = array(); - if (!is_null($oUser)) { - $oProfileSet = $oUser->Get('profile_list'); - while ($oProfile = $oProfileSet->Fetch()) { - $aUserProfiles[$oProfile->Get('profile')] = true; - } - } - - foreach ($aAllowedProfiles as $sProfile) { - if (array_key_exists($sProfile, $aUserProfiles)) { - $bCanKill = true; - break; - } - } - - if ($bCanKill) { - $this->AddMenuSeparator($aActions); - $aActions['concurrent_lock_unlock'] = array('label' => Dict::S('UI:Menu:KillConcurrentLock'), 'url' => "{$sRootUrl}pages/$sUIPage?operation=kill_lock&class=$sClass&id=$id{$sContext}"); - } - } } - $this->AddMenuSeparator($aActions); - /** @var \iApplicationUIExtension $oExtensionInstance */ - foreach (MetaModel::EnumPlugins('iApplicationUIExtension') as $oExtensionInstance) { - $oSet->Rewind(); - foreach ($oExtensionInstance->EnumAllowedActions($oSet) as $sLabel => $sUrl) { - $aActions[$sLabel] = array('label' => $sLabel, 'url' => $sUrl) + $aActionParams; - } - } - } - break; - - default: - // Check rights - // New / Modify - $bIsModifyAllowed = UserRights::IsActionAllowed($sClass, UR_ACTION_MODIFY, $oSet) && ($oReflectionClass->IsSubclassOf('cmdbAbstractObject')); - $bIsBulkModifyAllowed = (!MetaModel::IsAbstract($sClass)) && UserRights::IsActionAllowed($sClass, UR_ACTION_BULK_MODIFY, $oSet) && ($oReflectionClass->IsSubclassOf('cmdbAbstractObject')); - $bIsBulkDeleteAllowed = UserRights::IsActionAllowed($sClass, UR_ACTION_BULK_DELETE, $oSet); - if (isset($aExtraParams['link_attr'])) - { - $id = $aExtraParams['object_id']; - $sTargetAttr = $aExtraParams['target_attr']; - $oAttDef = MetaModel::GetAttributeDef($sClass, $sTargetAttr); - $sTargetClass = $oAttDef->GetTargetClass(); - if ($bIsModifyAllowed) { $aActions['UI:Menu:Add'] = array ('label' => Dict::S('UI:Menu:Add'), 'url' => "{$sRootUrl}pages/$sUIPage?operation=modify_links&class=$sClass&link_attr=".$aExtraParams['link_attr']."&target_class=$sTargetClass&id=$id&addObjects=true{$sContext}") + $aActionParams; } - if ($bIsBulkModifyAllowed) { $aActions['UI:Menu:Manage'] = array ('label' => Dict::S('UI:Menu:Manage'), 'url' => "{$sRootUrl}pages/$sUIPage?operation=modify_links&class=$sClass&link_attr=".$aExtraParams['link_attr']."&target_class=$sTargetClass&id=$id{$sContext}") + $aActionParams; } - //if ($bIsBulkDeleteAllowed) { $aActions[] = array ('label' => 'Remove All...', 'url' => "#") + $aActionParams; } } - else - { - // many objects in the set, possible actions are: new / modify all / delete all - if ($bIsCreationAllowed) { $aActions['UI:Menu:New'] = array ('label' => Dict::S('UI:Menu:New'), 'url' => "{$sRootUrl}pages/$sUIPage?operation=new&class=$sClass{$sContext}{$sDefault}") + $aActionParams; } - if ($bIsBulkModifyAllowed) { $aActions['UI:Menu:ModifyAll'] = array ('label' => Dict::S('UI:Menu:ModifyAll'), 'url' => "{$sRootUrl}pages/$sUIPage?operation=select_for_modify_all&class=$sClass&filter=".urlencode($sFilter)."{$sContext}") + $aActionParams; } - if ($bIsBulkDeleteAllowed) { $aActions['UI:Menu:BulkDelete'] = array ('label' => Dict::S('UI:Menu:BulkDelete'), 'url' => "{$sRootUrl}pages/$sUIPage?operation=select_for_deletion&filter=".urlencode($sFilter)."{$sContext}") + $aActionParams; } - // Stimuli - $aStates = MetaModel::EnumStates($sClass); - // Do not perform time consuming computations if there are too may objects in the list - $iLimit = MetaModel::GetConfig()->Get('complex_actions_limit'); - - if ((count($aStates) > 0) && (($iLimit == 0) || ($oSet->CountWithLimit($iLimit + 1) < $iLimit))) - { - // Life cycle actions may be available... if all objects are in the same state - // - // Group by - $oGroupByExp = new FieldExpression(MetaModel::GetStateAttributeCode($sClass), $this->m_oFilter->GetClassAlias()); - $aGroupBy = array('__state__' => $oGroupByExp); - $aQueryParams = array(); - if (isset($aExtraParams['query_params'])) - { - $aQueryParams = $aExtraParams['query_params']; - } - $sSql = $this->m_oFilter->MakeGroupByQuery($aQueryParams, $aGroupBy); - $aRes = CMDBSource::QueryToArray($sSql); - if (count($aRes) == 1) - { - // All objects are in the same state... - $sState = $aRes[0]['__state__']; - $aTransitions = Metamodel::EnumTransitions($sClass, $sState); - if (count($aTransitions)) - { - $this->AddMenuSeparator($aActions); - $aStimuli = Metamodel::EnumStimuli($sClass); - foreach($aTransitions as $sStimulusCode => $aTransitionDef) - { - $oSet->Rewind(); - // As soon as the user rights implementation will browse the object set, - // then we might consider using OptimizeColumnLoad() here - $iActionAllowed = UserRights::IsStimulusAllowed($sClass, $sStimulusCode, $oSet); - $iActionAllowed = (get_class($aStimuli[$sStimulusCode]) == 'StimulusUserAction') ? $iActionAllowed : UR_ALLOWED_NO; - switch($iActionAllowed) - { - case UR_ALLOWED_YES: - case UR_ALLOWED_DEPENDS: - $aActions[$sStimulusCode] = array('label' => $aStimuli[$sStimulusCode]->GetLabel(), 'url' => "{$sRootUrl}pages/UI.php?operation=select_bulk_stimulus&stimulus=$sStimulusCode&state=$sState&class=$sClass&filter=".urlencode($sFilter)."{$sContext}") + $aActionParams; - break; - - default: - // Do nothing - } - } - } + $this->AddMenuSeparator($aActions); + /** @var \iApplicationUIExtension $oExtensionInstance */ + foreach (MetaModel::EnumPlugins('iApplicationUIExtension') as $oExtensionInstance) { + $oSet->Rewind(); + foreach ($oExtensionInstance->EnumAllowedActions($oSet) as $sLabel => $data) { + if (is_array($data)) { + // New plugins can provide javascript handlers via the 'onclick' property + //TODO: enable extension of different menus by checking the 'target' property ?? + $aActions[$sLabel] = [ + 'label' => $sLabel, + 'url' => isset($data['url']) ? $data['url'] : '#', + 'onclick' => isset($data['onclick']) ? $data['onclick'] : '' + ]; + } else { + // Backward compatibility with old plugins + $aActions[$sLabel] = ['label' => $sLabel, 'url' => $data] + $aActionParams; } } } - } - - $this->AddMenuSeparator($aActions); - /** @var \iApplicationUIExtension $oExtensionInstance */ - foreach (MetaModel::EnumPlugins('iApplicationUIExtension') as $oExtensionInstance) { - $oSet->Rewind(); - foreach ($oExtensionInstance->EnumAllowedActions($oSet) as $sLabel => $data) { - if (is_array($data)) { - // New plugins can provide javascript handlers via the 'onclick' property - //TODO: enable extension of different menus by checking the 'target' property ?? - $aActions[$sLabel] = ['label' => $sLabel, 'url' => isset($data['url']) ? $data['url'] : '#', 'onclick' => isset($data['onclick']) ? $data['onclick'] : '']; - } else { - // Backward compatibility with old plugins - $aActions[$sLabel] = ['label' => $sLabel, 'url' => $data] + $aActionParams; - } + if (empty($sRefreshAction) && $this->m_sStyle == 'list') { + //for the detail page this var is defined way beyond this line + $sRefreshAction = "window.location.reload();"; } + } else + { + //it's easier just display configure this list and MENU_OBJLIST_TOOLKIT } $param = null; $iMenuId = null; @@ -1809,7 +1925,7 @@ class MenuBlock extends DisplayBlock } if ($bToolkitMenu) { $sLabel = Dict::S('UI:ConfigureThisList'); - $aActions['iTop::ConfigureList'] = ['label' => $sLabel, 'url' => '#', 'onclick' => "$('#datatable_dlg_{$sId}').dialog('open');"]; + $aActions['iTop::ConfigureList'] = ['label' => $sLabel, 'url' => '#', 'onclick' => "$('#datatable_dlg_datatable_{$sId}').dialog('open');"]; utils::GetPopupMenuItems($oPage, iPopupMenuExtension::MENU_OBJLIST_TOOLKIT, $param, $aActions); } break; @@ -1876,10 +1992,6 @@ JS $oActionsBlock->AddSubBlock($oActionButton); } - if (empty($sRefreshAction) && $this->m_sStyle == 'list') { - //for the detail page this var is defined way beyond this line - $sRefreshAction = "window.location.reload();"; - } if (!$oPage->IsPrintableVersion() && ($sRefreshAction != '')) { $oActionButton = ButtonFactory::MakeAlternativeNeutral('', 'UI:Button:Refresh'); $oActionButton->SetIconClass('fas fa-sync') diff --git a/application/shortcut.class.inc.php b/application/shortcut.class.inc.php index 42d220053..1453966c9 100644 --- a/application/shortcut.class.inc.php +++ b/application/shortcut.class.inc.php @@ -15,6 +15,7 @@ // // You should have received a copy of the GNU Affero General Public License // along with iTop. If not, see +use Combodo\iTop\Application\UI\Component\DataTable\DataTableSettings; /** diff --git a/application/ui.linksdirectwidget.class.inc.php b/application/ui.linksdirectwidget.class.inc.php index de4803312..b11e353e2 100644 --- a/application/ui.linksdirectwidget.class.inc.php +++ b/application/ui.linksdirectwidget.class.inc.php @@ -184,7 +184,7 @@ class UILinksWidgetDirect // For security reasons: check that the "proposed" class is actually a subclass of the linked class // and that the current user is allowed to create objects of this class $sRealClass = ''; - $oPage->add('
'); + //$oPage->add('
'); $aSubClasses = MetaModel::EnumChildClasses($this->sLinkedClass, ENUM_CHILD_CLASSES_ALL); // Including the specified class itself $aPossibleClasses = array(); foreach($aSubClasses as $sCandidateClass) @@ -294,7 +294,7 @@ class UILinksWidgetDirect */ public function GetObjectsSelectionDlg($oPage, $oCurrentObj, $aAlreadyLinked, $aPrefillFormParam = array()) { - $oPage->add("
\n"); + //$oPage->add("
\n"); $oHiddenFilter = new DBObjectSearch($this->sLinkedClass); if (($oCurrentObj != null) && MetaModel::IsSameFamilyBranch($this->sLinkedClass, $this->sClass)) { @@ -355,7 +355,6 @@ class UILinksWidgetDirect    -
HTML ); } diff --git a/application/ui.linkswidget.class.inc.php b/application/ui.linkswidget.class.inc.php index cdf9a86e4..aab215339 100644 --- a/application/ui.linkswidget.class.inc.php +++ b/application/ui.linkswidget.class.inc.php @@ -24,6 +24,10 @@ * @license http://opensource.org/licenses/AGPL-3.0 */ +use Combodo\iTop\Application\UI\Component\DataTable\DataTableFactory; +use Combodo\iTop\Application\UI\Component\DataTable\StaticTable\FormTableRow\FormTableRow; +use Combodo\iTop\Renderer\BlockRenderer; + require_once(APPROOT.'application/displayblock.class.inc.php'); class UILinksWidget @@ -335,67 +339,30 @@ JS ); } - /** - * Display one row of the whole form - * - * @param WebPage $oP - * @param array $aConfig - * @param array $aRow - * @param int $iRowId - * - * @return string - */ - protected function DisplayFormRow(WebPage $oP, $aConfig, $aRow, $iRowId) - { - $sHtml = ''; - $sHtml .= "m_sAttCode}{$this->m_sNameSuffix}_row_$iRowId\">\n"; - foreach($aConfig as $sName=>$void) - { - $sHtml .= "".$aRow[$sName]."\n"; - } - $sHtml .= "\n"; - - return $sHtml; - } - /** * Display the table with the form for editing all the links at once + * * @param WebPage $oP The web page used for the output * @param array $aConfig The table's header configuration * @param array $aData The tabular data to be displayed + * * @return string Html fragment representing the form table + * @throws \ReflectionException + * @throws \Twig\Error\LoaderError + * @throws \Twig\Error\RuntimeError + * @throws \Twig\Error\SyntaxError */ protected function DisplayFormTable(WebPage $oP, $aConfig, $aData) { - $sHtml = "m_sAttCode}{$this->m_sNameSuffix}\" value=\"\">"; - $sHtml .= "\n"; - // Header - $sHtml .= "\n"; - $sHtml .= "\n"; - foreach($aConfig as $sName=>$aDef) - { - $sHtml .= "\n"; - } - $sHtml .= "\n"; - $sHtml .= "\n"; - - // Content - $sHtml .= "\n"; - $sEmptyRowStyle = ''; - if (count($aData) != 0) - { - $sEmptyRowStyle = 'style="display:none;"'; - } + $oTable = DataTableFactory::MakeForForm("{$this->m_sAttCode}{$this->m_sNameSuffix}", $aConfig); foreach ($aData as $iRowId => $aRow) { - $sHtml .= $this->DisplayFormRow($oP, $aConfig, $aRow, $iRowId); + $oRow = new FormTableRow("{$this->m_sAttCode}{$this->m_sNameSuffix}", $aConfig, $aRow, $iRowId); + $oTable->AddRow($oRow); } - $sHtml .= "m_sAttCode}{$this->m_sNameSuffix}_empty_row\">"; - $sHtml .= "\n"; - // Footer - $sHtml .= "
".$aDef['label']."
".Dict::S('UI:Message:EmptyList:UseAdd')."
\n"; + $sHtml = BlockRenderer::RenderBlockTemplates($oTable); return $sHtml; } @@ -463,6 +430,7 @@ JS $aForm[$key] = $this->GetFormRow($oPage, $oLinkedObj, $oCurrentLink, $aArgs, $oCurrentObj, $key, $bReadOnly); } $sHtmlValue .= $this->DisplayFormTable($oPage, $this->m_aTableConfig, $aForm); + // To prevent adding forms inside the main form $sHtmlValue .= "     m_sAttCode}{$this->m_sNameSuffix}_btnRemove\" type=\"button\" value=\"".Dict::S('UI:RemoveLinkedObjectsOf_Class')."\" onClick=\"oWidget{$this->m_iInputId}.RemoveSelected();\" >"; $sHtmlValue .= "   m_sAttCode}{$this->m_sNameSuffix}_btnAdd\" type=\"button\" value=\"".Dict::Format('UI:AddLinkedObjectsOf_Class', MetaModel::GetName($this->m_sRemoteClass))."\" onClick=\"oWidget{$this->m_iInputId}.AddObjects();\">m_sAttCode}{$this->m_sNameSuffix}_indicatorAdd\">\n"; @@ -513,13 +481,13 @@ JS */ public function GetObjectPickerDialog($oPage, $oCurrentObj, $sJson, $aAlreadyLinkedIds = array(), $aPrefillFormParam = array()) { - $oPage->add("
\n"); + //$oPage->add("
\n"); $oAlreadyLinkedFilter = new DBObjectSearch($this->m_sRemoteClass); if (!$this->m_bDuplicatesAllowed && count($aAlreadyLinkedIds) > 0) { $oAlreadyLinkedFilter->AddCondition('id', $aAlreadyLinkedIds, 'NOTIN'); $oAlreadyLinkedExpression = $oAlreadyLinkedFilter->GetCriteria(); - $sAlreadyLinkedExpression = $oAlreadyLinkedExpression->Render(); + $sAlreadyLinkedExpression = $oAlreadyLinkedExpression->RenderExpression(); } else { $sAlreadyLinkedExpression = ''; } @@ -537,7 +505,7 @@ JS array( 'menu' => false, 'result_list_outer_selector' => "SearchResultsToAdd_{$this->m_sAttCode}{$this->m_sNameSuffix}", - 'table_id' => 'add_'.$this->m_sAttCode, + 'table_id' => "add_{$this->m_sAttCode}{$this->m_sNameSuffix}", 'table_inner_id' => "ResultsToAdd_{$this->m_sAttCode}{$this->m_sNameSuffix}", 'selection_mode' => true, 'json' => $sJson, @@ -555,14 +523,13 @@ JS

{$sEmptyList}

-    +    + -
HTML ); - $oPage->add_ready_script("$('#dlg_{$this->m_sAttCode}{$this->m_sNameSuffix}').dialog({ width: $(window).width()*0.8, height: $(window).height()*0.8, autoOpen: false, modal: true, resizeStop: oWidget{$this->m_iInputId}.UpdateSizes });"); - $oPage->add_ready_script("$('#dlg_{$this->m_sAttCode}{$this->m_sNameSuffix}').dialog('option', {title:'".addslashes(Dict::Format('UI:AddObjectsOf_Class_LinkedWith_Class', MetaModel::GetName($this->m_sLinkedClass), MetaModel::GetName($this->m_sClass)))."'});"); + $oPage->add_ready_script("$('#dlg_{$this->m_sAttCode}{$this->m_sNameSuffix}').dialog({ width: $(window).width()*0.8, height: $(window).height()*0.8, title:'".addslashes(Dict::Format('UI:AddObjectsOf_Class_LinkedWith_Class', MetaModel::GetName($this->m_sLinkedClass), MetaModel::GetName($this->m_sClass)))."' , autoOpen: false, modal: true, resizeStop: oWidget{$this->m_iInputId}.UpdateSizes });"); $oPage->add_ready_script("$('#SearchFormToAdd_{$this->m_sAttCode}{$this->m_sNameSuffix} form').bind('submit.uilinksWizard', oWidget{$this->m_iInputId}.SearchObjectsToAdd);"); $oPage->add_ready_script("$('#SearchFormToAdd_{$this->m_sAttCode}{$this->m_sNameSuffix}').resize(oWidget{$this->m_iInputId}.UpdateSizes);"); } @@ -620,7 +587,8 @@ HTML if (is_object($oLinkedObj)) { $aRow = $this->GetFormRow($oP, $oLinkedObj, $iObjectId, array(), $oCurrentObj, $iAdditionId); // Not yet created link get negative Ids - $oP->add($this->DisplayFormRow($oP, $this->m_aTableConfig, $aRow, -$iAdditionId)); + $oRow = new FormTableRow("{$this->m_sAttCode}{$this->m_sNameSuffix}", $this->m_aTableConfig, $aRow, -$iAdditionId); + $oP->AddUiBlock($oRow); $iAdditionId++; } else diff --git a/application/ui.searchformforeignkeys.class.inc.php b/application/ui.searchformforeignkeys.class.inc.php index 80fdb0cb2..8fd25a329 100644 --- a/application/ui.searchformforeignkeys.class.inc.php +++ b/application/ui.searchformforeignkeys.class.inc.php @@ -40,7 +40,6 @@ class UISearchFormForeignKeys */ public function ShowModalSearchForeignKeys($oPage, $sTitle) { - $oPage->add("
\n"); $oFilter = new DBObjectSearch($this->m_sRemoteClass); @@ -65,9 +64,9 @@ class UISearchFormForeignKeys

{$sEmptyList}

-    +    + -
HTML ); diff --git a/application/utils.inc.php b/application/utils.inc.php index 372731f99..f8e06f8c8 100644 --- a/application/utils.inc.php +++ b/application/utils.inc.php @@ -1697,7 +1697,7 @@ class utils set_time_limit(0); // Compiling SASS $sCss = $oSass->compile($sSassContent); - set_time_limit($iCurrentMaxExecTime); + set_time_limit(intval($iCurrentMaxExecTime)); return $sCss; } diff --git a/core/attributedef.class.inc.php b/core/attributedef.class.inc.php index e3e8cb04c..a5a7e59ae 100644 --- a/core/attributedef.class.inc.php +++ b/core/attributedef.class.inc.php @@ -1273,6 +1273,15 @@ abstract class AttributeDefinition { return (string)$value; } + + /* + * return string + */ + public function GetRenderForDataTable(string $sClassAlias) :string + { + $sRenderFunction = "return data;"; + return $sRenderFunction; + } } class AttributeDashboard extends AttributeDefinition @@ -4906,6 +4915,11 @@ class AttributeEmailAddress extends AttributeString return ''.parent::GetAsHTML($sValue).''; } + public function GetRenderForDataTable(string $sClassAlias) :string + { + $sRenderFunction = "return ' '+data+'' ;"; + return $sRenderFunction; + } } /** @@ -4992,6 +5006,12 @@ class AttributePhoneNumber extends AttributeString return ''.parent::GetAsHTML($sValue).''; } + + public function GetRenderForDataTable(string $sClassAlias) :string + { + $sRenderFunction = "return ' '+data+'' ;"; + return $sRenderFunction; + } } /** @@ -6781,6 +6801,14 @@ class AttributeExternalKey extends AttributeDBFieldVoid return DBObject::MakeHyperLink($this->GetTargetClass(), $sValue); } + + public function GetRenderForDataTable(string $sClassAlias) :string + { + $oRemoteAtt = $this->GetFinalAttDef(); + $sTargetClass = $oRemoteAtt->GetTargetClass(); + $sRenderFunction = "return ''+row['".$sClassAlias."/".$this->GetCode()."_friendlyname']+'' ;"; + return $sRenderFunction; + } } /** diff --git a/core/bulkchange.class.inc.php b/core/bulkchange.class.inc.php index 976d894d5..0af44ae97 100644 --- a/core/bulkchange.class.inc.php +++ b/core/bulkchange.class.inc.php @@ -948,7 +948,7 @@ class BulkChange $iLoopTimeLimit = MetaModel::GetConfig()->Get('max_execution_time_per_loop'); foreach($this->m_aData as $iRow => $aRowData) { - set_time_limit($iLoopTimeLimit); + set_time_limit(intval($iLoopTimeLimit)); if (isset($aResult[$iRow]["__STATUS__"])) { // An issue at the earlier steps - skip the rest @@ -1067,13 +1067,13 @@ class BulkChange $iObj = $oObj->GetKey(); if (!in_array($iObj, $aVisited)) { - set_time_limit($iLoopTimeLimit); + set_time_limit(intval($iLoopTimeLimit)); $iRow++; $this->UpdateMissingObject($aResult, $iRow, $oObj, $oChange); } } } - set_time_limit($iPreviousTimeLimit); + set_time_limit(itval($iPreviousTimeLimit)); // Fill in the blanks - the result matrix is expected to be 100% complete // diff --git a/core/csvbulkexport.class.inc.php b/core/csvbulkexport.class.inc.php index bfb22cfcf..fad0c38f0 100644 --- a/core/csvbulkexport.class.inc.php +++ b/core/csvbulkexport.class.inc.php @@ -301,7 +301,7 @@ EOF } while($aRow = $oSet->FetchAssoc()) { - set_time_limit($iLoopTimeLimit); + set_time_limit(intval($iLoopTimeLimit)); $aData = array(); foreach($this->aStatusInfo['fields'] as $iCol => $aFieldSpec) { @@ -339,7 +339,7 @@ EOF // Restore original date & time formats AttributeDateTime::SetFormat($oPrevDateTimeFormat); AttributeDate::SetFormat($oPrevDateFormat); - set_time_limit($iPreviousTimeLimit); + set_time_limit(intval($iPreviousTimeLimit)); $this->aStatusInfo['position'] += $this->iChunkSize; if ($this->aStatusInfo['total'] == 0) { diff --git a/core/csvparser.class.inc.php b/core/csvparser.class.inc.php index fae9af6d1..86df52b52 100644 --- a/core/csvparser.class.inc.php +++ b/core/csvparser.class.inc.php @@ -135,7 +135,7 @@ class CSVParser // More time for the next row if ($this->m_iTimeLimitPerRow !== null) { - set_time_limit($this->m_iTimeLimitPerRow); + set_time_limit(intval($this->m_iTimeLimitPerRow)); } } protected function __AddCellTrimmed($c = null, $aFieldMap = null) @@ -194,7 +194,7 @@ class CSVParser { // Give some time for the first row $iTimeLimit = ini_get('max_execution_time'); - set_time_limit($this->m_iTimeLimitPerRow); + set_time_limit(intval($this->m_iTimeLimitPerRow)); } for($i = 0; $i <= $iDataLength ; $i++) { @@ -255,7 +255,7 @@ class CSVParser if ($iTimeLimit !== null) { // Restore the previous time limit - set_time_limit($iTimeLimit); + set_time_limit(intval($iTimeLimit)); } return $this->m_aDataSet; } diff --git a/core/dbobject.class.php b/core/dbobject.class.php index 4dc9eac96..6ab87299e 100644 --- a/core/dbobject.class.php +++ b/core/dbobject.class.php @@ -3599,7 +3599,7 @@ abstract class DBObject implements iDisplay // As a temporary fix: delete only the objects that are still to be deleted... if ($oToDelete->m_bIsInDB) { - set_time_limit($iLoopTimeLimit); + set_time_limit(intval($iLoopTimeLimit)); $oToDelete->DBDeleteSingleObject(); } } @@ -3614,13 +3614,13 @@ abstract class DBObject implements iDisplay foreach ($aData['attributes'] as $sRemoteExtKey => $aRemoteAttDef) { $oToUpdate->Set($sRemoteExtKey, $aData['values'][$sRemoteExtKey]); - set_time_limit($iLoopTimeLimit); + set_time_limit(intval($iLoopTimeLimit)); $oToUpdate->DBUpdate(); } } } - set_time_limit($iPreviousTimeLimit); + set_time_limit(intval($iPreviousTimeLimit)); return $oDeletionPlan; } @@ -4565,7 +4565,7 @@ abstract class DBObject implements iDisplay { foreach ($aPotentialDeletes as $sRemoteExtKey => $aData) { - set_time_limit($iLoopTimeLimit); + set_time_limit(intval($iLoopTimeLimit)); /** @var \AttributeExternalKey $oAttDef */ $oAttDef = $aData['attribute']; @@ -4597,7 +4597,7 @@ abstract class DBObject implements iDisplay } } } - set_time_limit($iPreviousTimeLimit); + set_time_limit(intval($iPreviousTimeLimit)); } /** diff --git a/core/dbsearch.class.php b/core/dbsearch.class.php index e59493d80..95b65fa16 100644 --- a/core/dbsearch.class.php +++ b/core/dbsearch.class.php @@ -603,11 +603,11 @@ abstract class DBSearch elseif (($iPos = strpos($sParam, '->')) !== false) { $sParamName = substr($sParam, 0, $iPos); - if (isset($aContextParams[$sParamName.'->object()'])) + if (isset($aContextParams[$sParamName.'->object()']) || isset($aContextParams[$sParamName])) { $sAttCode = substr($sParam, $iPos + 2); /** @var \DBObject $oObj */ - $oObj = $aContextParams[$sParamName.'->object()']; + $oObj = isset($aContextParams[$sParamName.'->object()']) ? $aContextParams[$sParamName.'->object()'] : $aContextParams[$sParamName]; if ($oObj->IsModified()) { if ($sAttCode == 'id') diff --git a/core/deletionplan.class.inc.php b/core/deletionplan.class.inc.php index 447b458f1..9cae22135 100644 --- a/core/deletionplan.class.inc.php +++ b/core/deletionplan.class.inc.php @@ -116,7 +116,7 @@ class DeletionPlan { foreach($aToUpdate as $iId => $aData) { - set_time_limit($iLoopTimeLimit); + set_time_limit(intval($iLoopTimeLimit)); $this->m_iToUpdate++; $oObject = $aData['to_reset']; @@ -142,7 +142,7 @@ class DeletionPlan } } } - set_time_limit($iPreviousTimeLimit); + set_time_limit(intval($iPreviousTimeLimit)); } public function GetIssues() diff --git a/core/displayablegraph.class.inc.php b/core/displayablegraph.class.inc.php index 06b405543..559fd0c02 100644 --- a/core/displayablegraph.class.inc.php +++ b/core/displayablegraph.class.inc.php @@ -874,7 +874,7 @@ class DisplayableGraph extends SimpleGraph $oNodesIter = new RelationTypeIterator($oGraph, 'Node'); foreach($oNodesIter as $oNode) { - set_time_limit($iLoopTimeLimit); + set_time_limit(intval($iLoopTimeLimit)); switch(get_class($oNode)) { case 'RelationObjectNode': @@ -921,7 +921,7 @@ class DisplayableGraph extends SimpleGraph $oEdgesIter = new RelationTypeIterator($oGraph, 'Edge'); foreach($oEdgesIter as $oEdge) { - set_time_limit($iLoopTimeLimit); + set_time_limit(intval($iLoopTimeLimit)); $oSourceNode = $oNewGraph->GetNode($oEdge->GetSourceNode()->GetId()); $oSinkNode = $oNewGraph->GetNode($oEdge->GetSinkNode()->GetId()); $oNewEdge = new DisplayableEdge($oNewGraph, $oEdge->GetId(), $oSourceNode, $oSinkNode); @@ -932,7 +932,7 @@ class DisplayableGraph extends SimpleGraph $aEdgeKeys = array(); foreach($oEdgesIter as $oEdge) { - set_time_limit($iLoopTimeLimit); + set_time_limit(intval($iLoopTimeLimit)); $sSourceId = $oEdge->GetSourceNode()->GetId(); $sSinkId = $oEdge->GetSinkNode()->GetId(); if ($sSourceId == $sSinkId) @@ -958,7 +958,7 @@ class DisplayableGraph extends SimpleGraph $oNodesIter = new RelationTypeIterator($oNewGraph, 'Node'); foreach($oNodesIter as $oNode) { - set_time_limit($iLoopTimeLimit); + set_time_limit(intval($iLoopTimeLimit)); if ($bDirectionDown && $oNode->GetProperty('source')) { $oNode->GroupSimilarNeighbours($oNewGraph, $iGroupingThreshold, true, $bDirectionDown); @@ -973,7 +973,7 @@ class DisplayableGraph extends SimpleGraph $iGroupIdx = 0; foreach($oIterator as $oNode) { - set_time_limit($iLoopTimeLimit); + set_time_limit(intval($iLoopTimeLimit)); if ($oNode instanceof DisplayableGroupNode) { if ($oNode->GetObjectCount() == 0) @@ -995,7 +995,7 @@ class DisplayableGraph extends SimpleGraph $aEdgeKeys = array(); foreach($oEdgesIter as $oEdge) { - set_time_limit($iLoopTimeLimit); + set_time_limit(intval($iLoopTimeLimit)); $sSourceId = $oEdge->GetSourceNode()->GetId(); $sSinkId = $oEdge->GetSinkNode()->GetId(); if ($sSourceId == $sSinkId) @@ -1017,7 +1017,7 @@ class DisplayableGraph extends SimpleGraph } } } - set_time_limit($iPreviousTimeLimit); + set_time_limit(intval($iPreviousTimeLimit)); return $oNewGraph; } @@ -1257,14 +1257,14 @@ class DisplayableGraph extends SimpleGraph $iLoopTimeLimit = MetaModel::GetConfig()->Get('max_execution_time_per_loop'); foreach($oIterator as $sId => $oEdge) { - set_time_limit($iLoopTimeLimit); + set_time_limit(intval($iLoopTimeLimit)); $oEdge->RenderAsPDF($oPdf, $this, $fScale, $aContextDefs); } $oIterator = new RelationTypeIterator($this, 'Node'); foreach($oIterator as $sId => $oNode) { - set_time_limit($iLoopTimeLimit); + set_time_limit(intval($iLoopTimeLimit)); $oNode->RenderAsPDF($oPdf, $this, $fScale, $aContextDefs); } diff --git a/core/excelbulkexport.class.inc.php b/core/excelbulkexport.class.inc.php index 1d4569397..c4998cd92 100644 --- a/core/excelbulkexport.class.inc.php +++ b/core/excelbulkexport.class.inc.php @@ -296,7 +296,7 @@ EOF $iLoopTimeLimit = MetaModel::GetConfig()->Get('max_execution_time_per_loop'); while($aRow = $oSet->FetchAssoc()) { - set_time_limit($iLoopTimeLimit); + set_time_limit(intval($iLoopTimeLimit)); $aData = array(); foreach($this->aStatusInfo['fields'] as $iCol => $aFieldSpec) { @@ -314,7 +314,7 @@ EOF fwrite($hFile, json_encode($aData)."\n"); $iCount++; } - set_time_limit($iPreviousTimeLimit); + set_time_limit(intval($iPreviousTimeLimit)); $this->aStatusInfo['position'] += $this->iChunkSize; if ($this->aStatusInfo['total'] == 0) { diff --git a/core/htmlbulkexport.class.inc.php b/core/htmlbulkexport.class.inc.php index 8f811bb14..aea53bf88 100644 --- a/core/htmlbulkexport.class.inc.php +++ b/core/htmlbulkexport.class.inc.php @@ -128,7 +128,7 @@ class HTMLBulkExport extends TabularBulkExport $iLoopTimeLimit = MetaModel::GetConfig()->Get('max_execution_time_per_loop'); while($aRow = $oSet->FetchAssoc()) { - set_time_limit($iLoopTimeLimit); + set_time_limit(intval($iLoopTimeLimit)); $oMainObj = $aRow[$sFirstAlias]; $sHilightClass = ''; if ($oMainObj) @@ -160,7 +160,7 @@ class HTMLBulkExport extends TabularBulkExport $sData .= ""; $iCount++; } - set_time_limit($iPreviousTimeLimit); + set_time_limit(intval($iPreviousTimeLimit)); $this->aStatusInfo['position'] += $this->iChunkSize; if ($this->aStatusInfo['total'] == 0) { diff --git a/core/relationgraph.class.inc.php b/core/relationgraph.class.inc.php index 3e3521767..4dde465f8 100644 --- a/core/relationgraph.class.inc.php +++ b/core/relationgraph.class.inc.php @@ -472,7 +472,7 @@ class RelationGraph extends SimpleGraph { do { - set_time_limit($iLoopTimeLimit); + set_time_limit(intval($iLoopTimeLimit)); $sObjectRef = RelationObjectNode::MakeId($oRelatedObj); $oRelatedNode = $this->GetNode($sObjectRef); @@ -501,7 +501,7 @@ class RelationGraph extends SimpleGraph while ($oRelatedObj = $oObjSet->Fetch()); } } - set_time_limit($iPreviousTimeLimit); + set_time_limit(intval($iPreviousTimeLimit)); } } } diff --git a/core/spreadsheetbulkexport.class.inc.php b/core/spreadsheetbulkexport.class.inc.php index a742bc780..8a31453a6 100644 --- a/core/spreadsheetbulkexport.class.inc.php +++ b/core/spreadsheetbulkexport.class.inc.php @@ -266,7 +266,7 @@ EOF $iLoopTimeLimit = MetaModel::GetConfig()->Get('max_execution_time_per_loop'); while($aRow = $oSet->FetchAssoc()) { - set_time_limit($iLoopTimeLimit); + set_time_limit(intval($iLoopTimeLimit)); $sData .= ""; foreach($this->aStatusInfo['fields'] as $iCol => $aFieldSpec) @@ -362,7 +362,7 @@ EOF $sData .= ""; $iCount++; } - set_time_limit($iPreviousTimeLimit); + set_time_limit(intval($iPreviousTimeLimit)); $this->aStatusInfo['position'] += $this->iChunkSize; if ($this->aStatusInfo['total'] == 0) { diff --git a/core/xmlbulkexport.class.inc.php b/core/xmlbulkexport.class.inc.php index a94ecef1e..6c5d6bdb6 100644 --- a/core/xmlbulkexport.class.inc.php +++ b/core/xmlbulkexport.class.inc.php @@ -144,7 +144,7 @@ class XMLBulkExport extends BulkExport while ($aObjects = $oSet->FetchAssoc()) { - set_time_limit($iLoopTimeLimit); + set_time_limit(intval($iLoopTimeLimit)); if (count($aAuthorizedClasses) > 1) { $sData .= "\n"; @@ -182,7 +182,7 @@ class XMLBulkExport extends BulkExport $iCount++; } - set_time_limit($iPreviousTimeLimit); + set_time_limit(intval($iPreviousTimeLimit)); $this->aStatusInfo['position'] += $this->iChunkSize; if ($this->aStatusInfo['total'] == 0) { diff --git a/css/backoffice/components/_all.scss b/css/backoffice/components/_all.scss index 5eded7c2d..2350e5f50 100644 --- a/css/backoffice/components/_all.scss +++ b/css/backoffice/components/_all.scss @@ -18,8 +18,11 @@ @import "input/all"; @import "title"; +@import "datatable"; @import "form"; @import "fieldset"; @import "field"; @import "toolbar"; -@import "richtext"; \ No newline at end of file +@import "richtext"; +@import "formtable"; +@import "formtablerow"; \ No newline at end of file diff --git a/css/backoffice/components/_datatable.scss b/css/backoffice/components/_datatable.scss new file mode 100644 index 000000000..518acf74d --- /dev/null +++ b/css/backoffice/components/_datatable.scss @@ -0,0 +1,118 @@ +.ibo-datatable { + padding-bottom: 2em; + padding-top: 2em; + + thead tr th, tbody tr td { + line-height: 30px; + padding-right: 1em; + padding-left: 1em; + @extend %ibo-font-ral-nor-100; + + a { + color: $ibo-color-primary-600; + } + } + + tbody tr:nth-child(odd) { + background-color: $ibo-color-secondary-500; + } +} + .dataTables_length + { + @extend %ibo-font-ral-med-100; + color: $ibo-color-grey-700; + + position: relative; + float: right; + padding-top: 0.755em; + padding-right: 0.5em; + } + .dataTables_info + { + @extend %ibo-font-ral-med-100; + color: $ibo-color-grey-700; + + position: relative; + float: right; + padding-top: 0.755em; + padding-right: 2em; + padding-left: 0em; + } + .select-info{ + padding-left: 2em; + } + + .pagination + { + color: $ibo-color-grey-800; + + left: 1.21%; + right: 1.21%; + top: 91.58%; + bottom: 4.85%; + .paginate_button { + margin-left: 1em; + margin-right: 1em; + } + } + .dataTables_paginate + { + /* position: relative;*/ + float: left; + padding-top: 0.755em; + padding-left: 2em; + } + .paginate_button { + box-sizing: border-box; + display: inline-block; + } + .paginate_button a{ + color: $ibo-color-grey-800; + @extend %ibo-font-ral-nor-100; + } + .paginate_button a:hover { + color: $ibo-color-blue-grey-200; + } + .paginate_button.active a { + color: $ibo-color-grey-900; + border: 2px solid var(--ibo-color-grey-300); + background-color: $ibo-color-grey-200; + } +.ibo-datatable thead tr th{ + position: relative; +} +.ibo-datatable thead tr th.sorting::after{ + font-family: "Font Awesome 5 Free"; font-weight: 900; content: "\f0dc"; + opacity: 0.3; + right: 1em; + position: absolute; +} +.ibo-datatable thead tr th.sorting_desc:after{ + font-family: "Font Awesome 5 Free"; font-weight: 900; content: "\f0d7"; + right: 1em; + position: absolute; +} +.ibo-datatable thead tr th.sorting_asc:after{ + font-family: "Font Awesome 5 Free"; font-weight: 900; content: "\f0d8"; + right: 1em; + position: absolute; +} +.ibo-datatable-toolbar { + /*position: relative;*/ + //height: 30px; +} +.ibo-criterion-area{ + font-size: $ibo-font-size-50 ; +} +.ibo-search-form{ + padding-top: $ibo-panel--spacing-top ; +} + +.ibo-form-group{ + position: fixed; +} + +.sf_more_criterion.opened > .sfm_content{ + position: fixed !important; + left: auto !important; +} diff --git a/css/backoffice/components/_formtable.scss b/css/backoffice/components/_formtable.scss new file mode 100644 index 000000000..2b1c120e8 --- /dev/null +++ b/css/backoffice/components/_formtable.scss @@ -0,0 +1,4 @@ +/*! + * copyright Copyright (C) 2010-2020 Combodo SARL + * license http://opensource.org/licenses/AGPL-3.0 + */ diff --git a/css/backoffice/components/_formtablerow.scss b/css/backoffice/components/_formtablerow.scss new file mode 100644 index 000000000..2b1c120e8 --- /dev/null +++ b/css/backoffice/components/_formtablerow.scss @@ -0,0 +1,4 @@ +/*! + * copyright Copyright (C) 2010-2020 Combodo SARL + * license http://opensource.org/licenses/AGPL-3.0 + */ diff --git a/css/backoffice/components/_panel.scss b/css/backoffice/components/_panel.scss index b8a9fc852..119d0f300 100644 --- a/css/backoffice/components/_panel.scss +++ b/css/backoffice/components/_panel.scss @@ -77,6 +77,7 @@ $ibo-panel-colors: ( .ibo-panel--header { display: flex; + align-items: stretch; } .ibo-panel--title { diff --git a/css/backoffice/components/_title.scss b/css/backoffice/components/_title.scss index de0b60378..64f4402e7 100644 --- a/css/backoffice/components/_title.scss +++ b/css/backoffice/components/_title.scss @@ -79,4 +79,30 @@ $ibo-title--status-dot--border-radius: $ibo-border-radius-full !default; color: map-get($aColors, 'secondary-color'); background-color: map-get($aColors, 'primary-color'); } +} + + + +.ibo-title-for-dashlet { + padding-top: 2em; +} +.ibo-title-for-dashlet--title { + @extend %ibo-font-ral-nor-350; +} + +.ibo-title-for-dashlet--content{ + background-color: $ibo-color-white-100; + border-radius: 5px; + border: 1px solid ; + border-color:$ibo-color-grey-400; + padding-bottom:1em; +} +.ibo-title-separator{ + border-radius: 5px 5px 0px 0px; + border-color:$ibo-color-blue-600; + color:$ibo-color-blue-600; + background-color:$ibo-color-blue-600; + border: 3px solid ; + margin:0; + padding:0; } \ No newline at end of file diff --git a/css/light-grey.scss b/css/light-grey.scss index 90c702926..fd90c9ba1 100644 --- a/css/light-grey.scss +++ b/css/light-grey.scss @@ -93,66 +93,66 @@ text-decoration: none; } - table.datatable { - width: 100%; - border: 0; - padding: 0; - } + //table.datatable { + // width: 100%; + // border: 0; + // padding: 0; + //} td.menucontainer { text-align: right; } - table.listResults { - padding: 0px; - border-top: 3px solid $frame-background-color; - border-left: 3px solid $frame-background-color; - border-bottom: 3px solid #e6e6e1; - border-right: 3px solid #e6e6e1; - width: 100%; - background-color: $white; - } + //table.listResults { + // padding: 0px; + // border-top: 3px solid $frame-background-color; + // border-left: 3px solid $frame-background-color; + // border-bottom: 3px solid #e6e6e1; + // border-right: 3px solid #e6e6e1; + // width: 100%; + // background-color: $white; + //} - table.listResults td { - padding: 2px; - } + //table.listResults td { + // padding: 2px; + //} - table.listResults td .view-image { + //table.listResults td .view-image { // Counteract the forced dimensions (usefull for displaying in the details view) - width: inherit !important; - height: inherit !important; + // width: inherit !important; + // height: inherit !important; - img { - max-width: 48px !important; - max-height: 48px !important; - display: block; - margin-left: auto; - margin-right: auto; - } - } + // img { + // max-width: 48px !important; + // max-height: 48px !important; + // display: block; + // margin-left: auto; + // margin-right: auto; + // } + //} - table.listResults > tbody > tr.selected > * { - } + //table.listResults > tbody > tr.selected > * { + //} - table.listResults > tbody > tr > * { - transition: background-color 400ms linear; - } + //table.listResults > tbody > tr > * { + // transition: background-color 400ms linear; + //} - table.listResults > tbody > tr:hover > * { - cursor: pointer; - } + //table.listResults > tbody > tr:hover > * { + // cursor: pointer; + //} - table.listResults > tbody > tr.selected:hover > * { + //table.listResults > tbody > tr.selected:hover > * { /* hover on lines is currently done toggling td.hover, and having a rule for links */ - background-color: $brand-primary-lightest; - color: $text-color; - } + // background-color: $brand-primary-lightest; + // color: $text-color; + //} - table.listResults > tbody > tr:hover > * { + //table.listResults > tbody > tr:hover > * { /* hover on lines is currently done toggling td.hover, and having a rule for links */ - background-color: $table-hover-background; - color: $text-color; - } + // background-color: $table-hover-background; + // color: $text-color; + //} .edit-image { .view-image { @@ -278,13 +278,13 @@ color: $text-color; } - th { - font-family: Tahoma, Verdana, Arial, Helvetica; - font-size: 8pt; - color: $complement-color; - height: 20px; - background: $frame-background-color bottom repeat-x; - } + //th { + // font-family: Tahoma, Verdana, Arial, Helvetica; + // font-size: 8pt; + // color: $complement-color; + // height: 20px; + // background: $frame-background-color bottom repeat-x; + //} th.header { cursor: pointer; @@ -987,14 +987,14 @@ } .search_form_handler { - position: relative; - z-index: 10; - width: 100%; - margin-left: auto; - margin-right: auto; - font-size: 12px; - text-align: left; /* To compensate .search_box:text-align */ - border: 1px solid $search-form-container-bg-color; + // position: relative; + // z-index: 10; + // width: 100%; + // margin-left: auto; + // margin-right: auto; + // font-size: 12px; + // text-align: left; /* To compensate .search_box:text-align */ + // border: 1px solid $search-form-container-bg-color; //transition: width 0.3s ease-in-out; /* Sizing reset */ @@ -1138,7 +1138,6 @@ background-color: $white; .sf_criterion_row { - position: relative;; &:not(:first-child) { margin-top: 20px; @@ -1166,6 +1165,11 @@ .sf_criterion_group { display: inline; + .sfc_fg_button, + .sfc_header { + border: 1px solid #E1E7EC; /* Must be equal to .search_form_criteria:margin-bottom + this:padding-bottom */ + border-radius: 3px; + } } } @@ -1198,7 +1202,6 @@ box-shadow: $box-shadow-regular; } - .sfc_form_group, .sfm_content { position: absolute; z-index: -1; @@ -2643,7 +2646,6 @@ } > .field_input_btn { - display: table-cell; width: 25px; padding-left: 0.4em; diff --git a/datamodels/2.x/combodo-db-tools/db_analyzer.class.inc.php b/datamodels/2.x/combodo-db-tools/db_analyzer.class.inc.php index 29772994e..c41c83219 100644 --- a/datamodels/2.x/combodo-db-tools/db_analyzer.class.inc.php +++ b/datamodels/2.x/combodo-db-tools/db_analyzer.class.inc.php @@ -40,7 +40,7 @@ class DatabaseAnalyzer { if (!is_null($this->iTimeLimitPerOperation)) { - set_time_limit($this->iTimeLimitPerOperation); + set_time_limit(intval($this->iTimeLimitPerOperation)); } $aWrongRecords = CMDBSource::QueryToArray($sSelWrongRecs); diff --git a/datamodels/2.x/itop-portal-base/portal/templates/layout.html.twig b/datamodels/2.x/itop-portal-base/portal/templates/layout.html.twig index f1c429f4d..e6123d6dc 100644 --- a/datamodels/2.x/itop-portal-base/portal/templates/layout.html.twig +++ b/datamodels/2.x/itop-portal-base/portal/templates/layout.html.twig @@ -31,12 +31,12 @@ {# - Bootstrap Datetime picker #} {# - Datatables #} - - - - - - + + + + + + {# - Font OpenSans #} {# - Font Combodo #} @@ -112,13 +112,13 @@ {# Moment.js with locales#} {# Datatables #} - - - - - - - + + + + + + + {# Export for Datatables #} diff --git a/datamodels/2.x/itop-structure/precompiled-themes/light-grey/main.css b/datamodels/2.x/itop-structure/precompiled-themes/light-grey/main.css index 2525610ec..f61ae977a 100644 --- a/datamodels/2.x/itop-structure/precompiled-themes/light-grey/main.css +++ b/datamodels/2.x/itop-structure/precompiled-themes/light-grey/main.css @@ -1,6 +1,6 @@ /* === SIGNATURE BEGIN === -{"variables":"d751713988987e9331980363e24189ce","stylesheets":{"css-variables":"732f501302c538f594fb3c668d85b46c","jqueryui":"a887ad1c00f29132beba3bf729b981c7","main":"b04966c35dbdc85a2cd96abb6cd5fd6a"},"imports":[],"images":{"css\/ui-lightness\/images\/ui-icons_222222_256x240.png":"3a3c5468f484f07ac4a320d9e22acb8c","css\/ui-lightness\/images\/ui-bg_diagonals-thick_20_666666_40x40.png":"4429d568c67d8dfeb9040273ea0fb8c4","css\/ui-lightness\/images\/ui-icons_E87C1E_256x240.png":"7003dd36cb2aa032c8ec871ce4d4e03d","css\/ui-lightness\/images\/ui-icons_1c94c4_256x240.png":"dbd693dc8e0ef04e90a2f7ac7b390086","css\/ui-lightness\/images\/ui-icons_F26522_256x240.png":"16278ec0c07270be571f4c2e97fcc10c","css\/ui-lightness\/images\/ui-bg_diagonals-thick_18_b81900_40x40.png":"e460a66d4b3e093fc651e62a236267cb","css\/ui-lightness\/images\/ui-icons_ffffff_256x240.png":"41612b0f4a034424f8321c9f824a94da","css\/ui-lightness\/images\/ui-icons_ffd27a_256x240.png":"dda1b6f694b0d196aefc66a1d6d758f6","images\/actions_right.png":"31c8906bd25d27b83a0a2466bf903462","images\/ac-background.gif":"76135f3697b41a15aed787cfd77776c7","images\/green-square.gif":"16ea9a497d72f5e66e4e8ea9ae08024e","images\/tv-item.gif":"719fe2d4566108e73162fb8868d3778c","images\/tv-collapsable.gif":"63a3351ea0d580797c9b8c386aa4f48b","images\/tv-expandable.gif":"a2d1af4128e4a798a7f3390b12a28574","images\/tv-item-last.gif":"2ae7e1d9972ce71e5caa65a086bc5b7e","images\/tv-collapsable-last.gif":"71acaa9d7c2616e9e8b7131a75ca65da","images\/tv-expandable-last.gif":"9d51036b3a8102742709da66789fd0f7","images\/red-header.gif":"c73b8765f0c8c3c183cb6a0c2bb0ec69","images\/green-header.gif":"0e22a09bb8051b2a274b3427ede62e82","images\/orange-header.gif":"ce1f93f0af64431771b4cbd6c99c567b","images\/calendar.png":"ab56e59af3c96ca661821257d376465e","images\/truncated.png":"c6f91108afe8159d417b4dc556cd3b2a","images\/plus.gif":"f00e1e6e1161f48608bb2bbc79b9948c","images\/minus.gif":"6d77c0c0c2f86b6995d1cdf78274eaab","images\/full-screen.png":"b541fadd3f1563856a4b44aeebd9d563","images\/indicator.gif":"03ce3dcc84af110e9da8699a841e5200","images\/delete.png":"93c047549c31a270a037840277cf59d3","images\/info-mini.png":"445c090ed777c5e6a08ac390fa896193","images\/ok.png":"f6973773335fd83d8d2875f9a3c925af","images\/error.png":"1af8a1041016f67669c5fd22dc88c82e","images\/eye-open-555.png":"9940f4e5b1248042c238e1924359fd5e","images\/eye-closed-555.png":"6ad3b0bae791bf61addc9d8ca80a642d","images\/eye-open-fff.png":"b7db2402d4d5c72314c25790a66150d4","images\/eye-closed-fff.png":"f9be7454dbb47b0e0bca3aa370ae7db5"}} +{"variables":"d751713988987e9331980363e24189ce","stylesheets":{"css-variables":"732f501302c538f594fb3c668d85b46c","jqueryui":"a887ad1c00f29132beba3bf729b981c7","main":"d5d2002c5bcf8fcf592d8e028bbed2fb"},"imports":[],"images":{"css\/ui-lightness\/images\/ui-icons_222222_256x240.png":"3a3c5468f484f07ac4a320d9e22acb8c","css\/ui-lightness\/images\/ui-bg_diagonals-thick_20_666666_40x40.png":"4429d568c67d8dfeb9040273ea0fb8c4","css\/ui-lightness\/images\/ui-icons_E87C1E_256x240.png":"7003dd36cb2aa032c8ec871ce4d4e03d","css\/ui-lightness\/images\/ui-icons_1c94c4_256x240.png":"dbd693dc8e0ef04e90a2f7ac7b390086","css\/ui-lightness\/images\/ui-icons_F26522_256x240.png":"16278ec0c07270be571f4c2e97fcc10c","css\/ui-lightness\/images\/ui-bg_diagonals-thick_18_b81900_40x40.png":"e460a66d4b3e093fc651e62a236267cb","css\/ui-lightness\/images\/ui-icons_ffffff_256x240.png":"41612b0f4a034424f8321c9f824a94da","css\/ui-lightness\/images\/ui-icons_ffd27a_256x240.png":"dda1b6f694b0d196aefc66a1d6d758f6","images\/actions_right.png":"31c8906bd25d27b83a0a2466bf903462","images\/ac-background.gif":"76135f3697b41a15aed787cfd77776c7","images\/green-square.gif":"16ea9a497d72f5e66e4e8ea9ae08024e","images\/tv-item.gif":"719fe2d4566108e73162fb8868d3778c","images\/tv-collapsable.gif":"63a3351ea0d580797c9b8c386aa4f48b","images\/tv-expandable.gif":"a2d1af4128e4a798a7f3390b12a28574","images\/tv-item-last.gif":"2ae7e1d9972ce71e5caa65a086bc5b7e","images\/tv-collapsable-last.gif":"71acaa9d7c2616e9e8b7131a75ca65da","images\/tv-expandable-last.gif":"9d51036b3a8102742709da66789fd0f7","images\/red-header.gif":"c73b8765f0c8c3c183cb6a0c2bb0ec69","images\/green-header.gif":"0e22a09bb8051b2a274b3427ede62e82","images\/orange-header.gif":"ce1f93f0af64431771b4cbd6c99c567b","images\/calendar.png":"ab56e59af3c96ca661821257d376465e","images\/truncated.png":"c6f91108afe8159d417b4dc556cd3b2a","images\/plus.gif":"f00e1e6e1161f48608bb2bbc79b9948c","images\/minus.gif":"6d77c0c0c2f86b6995d1cdf78274eaab","images\/full-screen.png":"b541fadd3f1563856a4b44aeebd9d563","images\/indicator.gif":"03ce3dcc84af110e9da8699a841e5200","images\/delete.png":"93c047549c31a270a037840277cf59d3","images\/info-mini.png":"445c090ed777c5e6a08ac390fa896193","images\/ok.png":"f6973773335fd83d8d2875f9a3c925af","images\/error.png":"1af8a1041016f67669c5fd22dc88c82e","images\/eye-open-555.png":"9940f4e5b1248042c238e1924359fd5e","images\/eye-closed-555.png":"6ad3b0bae791bf61addc9d8ca80a642d","images\/eye-open-fff.png":"b7db2402d4d5c72314c25790a66150d4","images\/eye-closed-fff.png":"f9be7454dbb47b0e0bca3aa370ae7db5"}} === SIGNATURE END === */ /*! @@ -99,7 +99,7 @@ background-repeat: no-repeat; width: 16px; height: 16px; - background-image: url("../../../../css/ui-lightness/images/ui-icons_222222_256x240.png?v=1605021905"); + background-image: url("../../../../css/ui-lightness/images/ui-icons_222222_256x240.png?v=1605536038"); filter: hue-rotate(0deg); } .ui-widget-icon-block { @@ -232,7 +232,7 @@ .ui-menu .ui-menu-item { margin: 0; cursor: pointer; - list-style-image: url("?v=1605021905"); + list-style-image: url("?v=1605536038"); } .ui-menu .ui-menu-item-wrapper { position: relative; @@ -298,7 +298,7 @@ color: #EA7D1E; } .ui-button:hover .ui-icon { - background-image: url("../../../../css/ui-lightness/images/ui-icons_E87C1E_256x240.png?v=1605021905"); + background-image: url("../../../../css/ui-lightness/images/ui-icons_E87C1E_256x240.png?v=1605536038"); } .ui-button:active { text-decoration: none; @@ -306,20 +306,20 @@ color: #EA7D1E; } .ui-button:active .ui-icon { - background-image: url("../../../../css/ui-lightness/images/ui-icons_E87C1E_256x240.png?v=1605021905"); + background-image: url("../../../../css/ui-lightness/images/ui-icons_E87C1E_256x240.png?v=1605536038"); } .ui-button:focus { font-weight: bold; color: #EA7D1E; } .ui-button:focus .ui-icon { - background-image: url("../../../../css/ui-lightness/images/ui-icons_E87C1E_256x240.png?v=1605021905"); + background-image: url("../../../../css/ui-lightness/images/ui-icons_E87C1E_256x240.png?v=1605536038"); } .ui-button .ui-state-highlight.ui-icon { - background-image: url("../../../../css/ui-lightness/images/ui-icons_1c94c4_256x240.png?v=1605021905"); + background-image: url("../../../../css/ui-lightness/images/ui-icons_1c94c4_256x240.png?v=1605536038"); } .ui-button .ui-icon { - background-image: url("../../../../css/ui-lightness/images/ui-icons_F26522_256x240.png?v=1605021905"); + background-image: url("../../../../css/ui-lightness/images/ui-icons_F26522_256x240.png?v=1605536038"); } .ui-button-icon-only { width: 2em; @@ -901,7 +901,7 @@ body .ui-tooltip { background-image: none; } .ui-widget-content .ui-icon { - background-image: url("../../../../css/ui-lightness/images/ui-icons_222222_256x240.png?v=1605021905"); + background-image: url("../../../../css/ui-lightness/images/ui-icons_222222_256x240.png?v=1605536038"); } .ui-widget-header .ui-priority-secondary { opacity: 0.7; @@ -913,7 +913,7 @@ body .ui-tooltip { background-image: none; } .ui-widget-header .ui-icon { - background-image: url("../../../../css/ui-lightness/images/ui-icons_ffffff_256x240.png?v=1605021905"); + background-image: url("../../../../css/ui-lightness/images/ui-icons_ffffff_256x240.png?v=1605536038"); } .ui-state-default a { text-decoration: none; @@ -925,7 +925,7 @@ body .ui-tooltip { text-decoration: none; } .ui-state-default .ui-icon { - background-image: url("../../../../css/ui-lightness/images/ui-icons_F26522_256x240.png?v=1605021905"); + background-image: url("../../../../css/ui-lightness/images/ui-icons_F26522_256x240.png?v=1605536038"); } a.ui-button { text-decoration: none; @@ -955,7 +955,7 @@ a:visited.ui-button { text-decoration: none; } .ui-state-hover .ui-icon { - background-image: url("../../../../css/ui-lightness/images/ui-icons_E87C1E_256x240.png?v=1605021905"); + background-image: url("../../../../css/ui-lightness/images/ui-icons_E87C1E_256x240.png?v=1605536038"); } .ui-state-focus a { text-decoration: none; @@ -970,7 +970,7 @@ a:visited.ui-button { text-decoration: none; } .ui-state-focus .ui-icon { - background-image: url("../../../../css/ui-lightness/images/ui-icons_E87C1E_256x240.png?v=1605021905"); + background-image: url("../../../../css/ui-lightness/images/ui-icons_E87C1E_256x240.png?v=1605536038"); } .ui-visual-focus { box-shadow: 0 0 3px 1px #5e9ed6; @@ -988,19 +988,19 @@ a:visited.ui-button { text-decoration: none; } .ui-state-active .ui-icon { - background-image: url("../../../../css/ui-lightness/images/ui-icons_E87C1E_256x240.png?v=1605021905"); + background-image: url("../../../../css/ui-lightness/images/ui-icons_E87C1E_256x240.png?v=1605536038"); } .ui-icon-background { background-color: #EA7D1E; } .ui-state-highlight .ui-icon { - background-image: url("../../../../css/ui-lightness/images/ui-icons_1c94c4_256x240.png?v=1605021905"); + background-image: url("../../../../css/ui-lightness/images/ui-icons_1c94c4_256x240.png?v=1605536038"); } .ui-state-error .ui-icon { - background-image: url("../../../../css/ui-lightness/images/ui-icons_ffd27a_256x240.png?v=1605021905"); + background-image: url("../../../../css/ui-lightness/images/ui-icons_ffd27a_256x240.png?v=1605536038"); } .ui-state-error-text .ui-icon { - background-image: url("../../../../css/ui-lightness/images/ui-icons_ffd27a_256x240.png?v=1605021905"); + background-image: url("../../../../css/ui-lightness/images/ui-icons_ffd27a_256x240.png?v=1605536038"); } .ui-priority-primary { font-weight: bold; @@ -1643,52 +1643,10 @@ a:visited.ui-button { color: #EA7D1E; text-decoration: none; } -#ibo-main-content table.datatable, .ui-dialog table.datatable { - width: 100%; - border: 0; - padding: 0; -} #ibo-main-content td.menucontainer, .ui-dialog td.menucontainer { text-align: right; -} -#ibo-main-content table.listResults, .ui-dialog table.listResults { - padding: 0px; - border-top: 3px solid #F1F1F1; - border-left: 3px solid #F1F1F1; - border-bottom: 3px solid #e6e6e1; - border-right: 3px solid #e6e6e1; - width: 100%; - background-color: #FFFFFF; -} -#ibo-main-content table.listResults td, .ui-dialog table.listResults td { - padding: 2px; -} -#ibo-main-content table.listResults td .view-image, .ui-dialog table.listResults td .view-image { - width: inherit !important; - height: inherit !important; -} -#ibo-main-content table.listResults td .view-image img, .ui-dialog table.listResults td .view-image img { - max-width: 48px !important; - max-height: 48px !important; - display: block; - margin-left: auto; - margin-right: auto; -} -#ibo-main-content table.listResults > tbody > tr > *, .ui-dialog table.listResults > tbody > tr > * { - transition: background-color 400ms linear; -} -#ibo-main-content table.listResults > tbody > tr:hover > *, .ui-dialog table.listResults > tbody > tr:hover > * { - cursor: pointer; -} -#ibo-main-content table.listResults > tbody > tr.selected:hover > *, .ui-dialog table.listResults > tbody > tr.selected:hover > * { /* hover on lines is currently done toggling td.hover, and having a rule for links */ - background-color: #f1a564; - color: #000; -} -#ibo-main-content table.listResults > tbody > tr:hover > *, .ui-dialog table.listResults > tbody > tr:hover > * { /* hover on lines is currently done toggling td.hover, and having a rule for links */ - background-color: #fdf5d0; - color: #000; } #ibo-main-content .edit-image .view-image, .ui-dialog .edit-image .view-image { display: inline-block; @@ -1697,7 +1655,7 @@ a:visited.ui-button { visibility: hidden; } #ibo-main-content .edit-image .view-image.dirty.compat, .ui-dialog .edit-image .view-image.dirty.compat { - background-image: url("../../../../css/ui-lightness/images/ui-bg_diagonals-thick_20_666666_40x40.png?v=1605021905"); + background-image: url("../../../../css/ui-lightness/images/ui-bg_diagonals-thick_20_666666_40x40.png?v=1605536038"); } #ibo-main-content .edit-image .view-image.dirty.compat img, .ui-dialog .edit-image .view-image.dirty.compat img { opacity: 0.3; @@ -1720,7 +1678,7 @@ a:visited.ui-button { opacity: 0.3; } #ibo-main-content .edit-image .edit-buttons .button .ui-icon, .ui-dialog .edit-image .edit-buttons .button .ui-icon { - background-image: url("../../../../css/ui-lightness/images/ui-icons_ffffff_256x240.png?v=1605021905"); + background-image: url("../../../../css/ui-lightness/images/ui-icons_ffffff_256x240.png?v=1605536038"); } #ibo-main-content .edit-image .file-input, .ui-dialog .edit-image .file-input { display: block; @@ -1781,13 +1739,6 @@ a:visited.ui-button { #ibo-main-content tr td.hover, .ui-dialog tr td.hover, #ibo-main-content tr.even td.hover, .ui-dialog tr.even td.hover, #ibo-main-content .hover a, .ui-dialog .hover a, #ibo-main-content .hover a:visited, .ui-dialog .hover a:visited, #ibo-main-content .hover a:hover, .ui-dialog .hover a:hover, #ibo-main-content .wizContainer tr.even td.hover, .ui-dialog .wizContainer tr.even td.hover, #ibo-main-content .wizContainer tr td.hover, .ui-dialog .wizContainer tr td.hover { color: #000; } -#ibo-main-content th, .ui-dialog th { - font-family: Tahoma, Verdana, Arial, Helvetica; - font-size: 8pt; - color: #1c94c4; - height: 20px; - background: #F1F1F1 bottom repeat-x; -} #ibo-main-content th.header, .ui-dialog th.header { cursor: pointer; background-repeat: no-repeat; @@ -1940,7 +1891,7 @@ a:visited.ui-button { background: #EA7D1E; } #ibo-main-content .actions_details span, .ui-dialog .actions_details span { - background: url("../../../../images/actions_right.png?v=1605021905") no-repeat right; + background: url("../../../../images/actions_right.png?v=1605536038") no-repeat right; color: #fff; font-weight: bold; padding-top: 2px; @@ -1963,7 +1914,7 @@ a:visited.ui-button { } #ibo-main-content .ac_input, .ui-dialog .ac_input { border: 1px solid #7f9db9; - background: #fff url("../../../../images/ac-background.gif?v=1605021905") no-repeat right; + background: #fff url("../../../../images/ac-background.gif?v=1605536038") no-repeat right; /* By Rom */ } #ibo-main-content .csvimport_createobj, .ui-dialog .csvimport_createobj { @@ -2018,7 +1969,7 @@ a:visited.ui-button { noborder-top: 1px solid #8b8b8b; padding: 4px 0px 0px 16px; font-size: 8pt; - background: url("../../../../images/green-square.gif?v=1605021905") no-repeat bottom left; + background: url("../../../../images/green-square.gif?v=1605536038") no-repeat bottom left; color: #83b217; font-weight: bold; text-decoration: none; @@ -2094,22 +2045,22 @@ a:visited.ui-button { margin-top: -8px; } #ibo-main-content .notreeview li, .ui-dialog .notreeview li { - background: url("../../../../images/tv-item.gif?v=1605021905") 0 0 no-repeat; + background: url("../../../../images/tv-item.gif?v=1605536038") 0 0 no-repeat; } #ibo-main-content .notreeview .collapsable, .ui-dialog .notreeview .collapsable { - background-image: url("../../../../images/tv-collapsable.gif?v=1605021905"); + background-image: url("../../../../images/tv-collapsable.gif?v=1605536038"); } #ibo-main-content .notreeview .expandable, .ui-dialog .notreeview .expandable { - background-image: url("../../../../images/tv-expandable.gif?v=1605021905"); + background-image: url("../../../../images/tv-expandable.gif?v=1605536038"); } #ibo-main-content .notreeview .last, .ui-dialog .notreeview .last { - background-image: url("../../../../images/tv-item-last.gif?v=1605021905"); + background-image: url("../../../../images/tv-item-last.gif?v=1605536038"); } #ibo-main-content .notreeview .lastCollapsable, .ui-dialog .notreeview .lastCollapsable { - background-image: url("../../../../images/tv-collapsable-last.gif?v=1605021905"); + background-image: url("../../../../images/tv-collapsable-last.gif?v=1605536038"); } #ibo-main-content .notreeview .lastExpandable, .ui-dialog .notreeview .lastExpandable { - background-image: url("../../../../images/tv-expandable-last.gif?v=1605021905"); + background-image: url("../../../../images/tv-expandable-last.gif?v=1605536038"); } #ibo-main-content #OrganizationSelection, .ui-dialog #OrganizationSelection { padding: 5px 0px 16px 20px; @@ -2285,7 +2236,7 @@ a:visited.ui-button { color: #000; } #ibo-main-content th.red, .ui-dialog th.red { - background: url("../../../../images/red-header.gif?v=1605021905") bottom left repeat-x; + background: url("../../../../images/red-header.gif?v=1605536038") bottom left repeat-x; color: #000; } #ibo-main-content .green, .ui-dialog .green { @@ -2293,7 +2244,7 @@ a:visited.ui-button { color: #000; } #ibo-main-content th.green, .ui-dialog th.green { - background: url("../../../../images/green-header.gif?v=1605021905") bottom left repeat-x; + background: url("../../../../images/green-header.gif?v=1605536038") bottom left repeat-x; color: #000; } #ibo-main-content .orange, .ui-dialog .orange { @@ -2301,7 +2252,7 @@ a:visited.ui-button { color: #000; } #ibo-main-content th.orange, .ui-dialog th.orange { - background: url("../../../../images/orange-header.gif?v=1605021905") bottom left repeat-x; + background: url("../../../../images/orange-header.gif?v=1605536038") bottom left repeat-x; color: #000; /* For Date Picker: Creates a little calendar icon * instead of a text link for "Choose date" @@ -2316,7 +2267,7 @@ a:visited.ui-button { display: block; text-indent: -2000px; overflow: hidden; - background: url("../../../../images/calendar.png?v=1605021905") no-repeat; + background: url("../../../../images/calendar.png?v=1605536038") no-repeat; } #ibo-main-content td a.dp-choose-date.dp-disabled, .ui-dialog td a.dp-choose-date.dp-disabled, #ibo-main-content a.dp-choose-date.dp-disabled, .ui-dialog a.dp-choose-date.dp-disabled { background-position: 0 -20px; @@ -2368,15 +2319,6 @@ a:visited.ui-button { box-sizing: border-box; } #ibo-main-content .search_form_handler, .ui-dialog .search_form_handler { - position: relative; - z-index: 10; - width: 100%; - margin-left: auto; - margin-right: auto; - font-size: 12px; - text-align: left; - /* To compensate .search_box:text-align */ - border: 1px solid #1c94c4; /* Sizing reset */ } #ibo-main-content .search_form_handler *, .ui-dialog .search_form_handler * { @@ -2475,9 +2417,6 @@ a:visited.ui-button { /* padding-bottom must equals to padding-top - .search_form_criteria:margin-bottom */ background-color: #FFFFFF; } -#ibo-main-content .search_form_handler .sf_criterion_area .sf_criterion_row, .ui-dialog .search_form_handler .sf_criterion_area .sf_criterion_row { - position: relative; -} #ibo-main-content .search_form_handler .sf_criterion_area .sf_criterion_row:not(:first-child), .ui-dialog .search_form_handler .sf_criterion_area .sf_criterion_row:not(:first-child) { margin-top: 20px; } @@ -2503,6 +2442,11 @@ a:visited.ui-button { } #ibo-main-content .search_form_handler .sf_criterion_area .sf_criterion_row .sf_criterion_group, .ui-dialog .search_form_handler .sf_criterion_area .sf_criterion_row .sf_criterion_group { display: inline; +} +#ibo-main-content .search_form_handler .sf_criterion_area .sf_criterion_row .sf_criterion_group .sfc_fg_button, .ui-dialog .search_form_handler .sf_criterion_area .sf_criterion_row .sf_criterion_group .sfc_fg_button, #ibo-main-content .search_form_handler .sf_criterion_area .sf_criterion_row .sf_criterion_group .sfc_header, .ui-dialog .search_form_handler .sf_criterion_area .sf_criterion_row .sf_criterion_group .sfc_header { + border: 1px solid #E1E7EC; + /* Must be equal to .search_form_criteria:margin-bottom + this:padding-bottom */ + border-radius: 3px; /* Common style between criterion and more criterion */ } #ibo-main-content .search_form_handler .sf_criterion_area .search_form_criteria, .ui-dialog .search_form_handler .sf_criterion_area .search_form_criteria, #ibo-main-content .search_form_handler .sf_criterion_area .sf_more_criterion, .ui-dialog .search_form_handler .sf_criterion_area .sf_more_criterion, #ibo-main-content .search_form_handler .sf_criterion_area .sf_button, .ui-dialog .search_form_handler .sf_criterion_area .sf_button { @@ -2529,14 +2473,14 @@ a:visited.ui-button { border-radius: 1px; box-shadow: 0 1px 1px rgba(0, 0, 0, 0.15); } -#ibo-main-content .search_form_handler .sf_criterion_area .search_form_criteria .sfc_form_group, .ui-dialog .search_form_handler .sf_criterion_area .search_form_criteria .sfc_form_group, #ibo-main-content .search_form_handler .sf_criterion_area .sf_more_criterion .sfc_form_group, .ui-dialog .search_form_handler .sf_criterion_area .sf_more_criterion .sfc_form_group, #ibo-main-content .search_form_handler .sf_criterion_area .sf_button .sfc_form_group, .ui-dialog .search_form_handler .sf_criterion_area .sf_button .sfc_form_group, #ibo-main-content .search_form_handler .sf_criterion_area .search_form_criteria .sfm_content, .ui-dialog .search_form_handler .sf_criterion_area .search_form_criteria .sfm_content, #ibo-main-content .search_form_handler .sf_criterion_area .sf_more_criterion .sfm_content, .ui-dialog .search_form_handler .sf_criterion_area .sf_more_criterion .sfm_content, #ibo-main-content .search_form_handler .sf_criterion_area .sf_button .sfm_content, .ui-dialog .search_form_handler .sf_criterion_area .sf_button .sfm_content { +#ibo-main-content .search_form_handler .sf_criterion_area .search_form_criteria .sfm_content, .ui-dialog .search_form_handler .sf_criterion_area .search_form_criteria .sfm_content, #ibo-main-content .search_form_handler .sf_criterion_area .sf_more_criterion .sfm_content, .ui-dialog .search_form_handler .sf_criterion_area .sf_more_criterion .sfm_content, #ibo-main-content .search_form_handler .sf_criterion_area .sf_button .sfm_content, .ui-dialog .search_form_handler .sf_criterion_area .sf_button .sfm_content { position: absolute; z-index: -1; min-width: 100%; left: 0px; margin-top: -1px; } -#ibo-main-content .search_form_handler .sf_criterion_area .search_form_criteria .sfc_form_group .sfc_fg_buttons, .ui-dialog .search_form_handler .sf_criterion_area .search_form_criteria .sfc_form_group .sfc_fg_buttons, #ibo-main-content .search_form_handler .sf_criterion_area .sf_more_criterion .sfc_form_group .sfc_fg_buttons, .ui-dialog .search_form_handler .sf_criterion_area .sf_more_criterion .sfc_form_group .sfc_fg_buttons, #ibo-main-content .search_form_handler .sf_criterion_area .sf_button .sfc_form_group .sfc_fg_buttons, .ui-dialog .search_form_handler .sf_criterion_area .sf_button .sfc_form_group .sfc_fg_buttons, #ibo-main-content .search_form_handler .sf_criterion_area .search_form_criteria .sfm_content .sfc_fg_buttons, .ui-dialog .search_form_handler .sf_criterion_area .search_form_criteria .sfm_content .sfc_fg_buttons, #ibo-main-content .search_form_handler .sf_criterion_area .sf_more_criterion .sfm_content .sfc_fg_buttons, .ui-dialog .search_form_handler .sf_criterion_area .sf_more_criterion .sfm_content .sfc_fg_buttons, #ibo-main-content .search_form_handler .sf_criterion_area .sf_button .sfm_content .sfc_fg_buttons, .ui-dialog .search_form_handler .sf_criterion_area .sf_button .sfm_content .sfc_fg_buttons { +#ibo-main-content .search_form_handler .sf_criterion_area .search_form_criteria .sfm_content .sfc_fg_buttons, .ui-dialog .search_form_handler .sf_criterion_area .search_form_criteria .sfm_content .sfc_fg_buttons, #ibo-main-content .search_form_handler .sf_criterion_area .sf_more_criterion .sfm_content .sfc_fg_buttons, .ui-dialog .search_form_handler .sf_criterion_area .sf_more_criterion .sfm_content .sfc_fg_buttons, #ibo-main-content .search_form_handler .sf_criterion_area .sf_button .sfm_content .sfc_fg_buttons, .ui-dialog .search_form_handler .sf_criterion_area .sf_button .sfm_content .sfc_fg_buttons { white-space: nowrap; /* Criteria tags */ } @@ -2983,19 +2927,19 @@ a:visited.ui-button { /* Beware: IE6 does not support multiple selector with multiple classes, only the last class is used */ } #ibo-main-content table.listResults tr.odd td.truncated, .ui-dialog table.listResults tr.odd td.truncated, #ibo-main-content table.listResults tr td.truncated, .ui-dialog table.listResults tr td.truncated, #ibo-main-content .wizContainer table.listResults tr.odd td.truncated, .ui-dialog .wizContainer table.listResults tr.odd td.truncated, #ibo-main-content .wizContainer table.listResults tr td.truncated, .ui-dialog .wizContainer table.listResults tr td.truncated { - background: url("../../../../images/truncated.png?v=1605021905") bottom repeat-x; + background: url("../../../../images/truncated.png?v=1605536038") bottom repeat-x; /* Beware: IE6 does not support multiple selector with multiple classes, only the last class is used */ } #ibo-main-content table.listResults tr.even td.truncated, .ui-dialog table.listResults tr.even td.truncated, #ibo-main-content .wizContainer table.listResults tr.even td.truncated, .ui-dialog .wizContainer table.listResults tr.even td.truncated { - background: #f9f9f1 url("../../../../images/truncated.png?v=1605021905") bottom repeat-x; + background: #f9f9f1 url("../../../../images/truncated.png?v=1605536038") bottom repeat-x; /* Beware: IE6 does not support multiple selector with multiple classes, only the last class is used */ } #ibo-main-content table.listResults tr.even td.hover.truncated, .ui-dialog table.listResults tr.even td.hover.truncated, #ibo-main-content .wizContainer table.listResults tr.even td.hover.truncated, .ui-dialog .wizContainer table.listResults tr.even td.hover.truncated { - background: #fdf5d0 url("../../../../images/truncated.png?v=1605021905") bottom repeat-x; + background: #fdf5d0 url("../../../../images/truncated.png?v=1605536038") bottom repeat-x; /* Beware: IE6 does not support multiple selector with multiple classes, only the last class is used */ } #ibo-main-content table.listResults tr.odd td.hover.truncated, .ui-dialog table.listResults tr.odd td.hover.truncated, #ibo-main-content table.listResults tr td.hover.truncated, .ui-dialog table.listResults tr td.hover.truncated, #ibo-main-content .wizContainer table.listResults tr.odd td.hover.truncated, .ui-dialog .wizContainer table.listResults tr.odd td.hover.truncated, #ibo-main-content .wizContainer table.listResults tr td.hover.truncated, .ui-dialog .wizContainer table.listResults tr td.hover.truncated { - background: #fdf5d0 url("../../../../images/truncated.png?v=1605021905") bottom repeat-x; + background: #fdf5d0 url("../../../../images/truncated.png?v=1605536038") bottom repeat-x; } #ibo-main-content table.listResults.truncated, .ui-dialog table.listResults.truncated { border-bottom: 0; @@ -3097,7 +3041,7 @@ a:visited.ui-button { #ibo-main-content div#logo div, .ui-dialog div#logo div { height: 88px; width: 244px; - background: url("../../../../images/itop-logo-2.png?v=1605021905") left no-repeat; + background: url("../../../../images/itop-logo-2.png?v=1605536038") left no-repeat; } #ibo-main-content #left-pane .ui-layout-north, .ui-dialog #left-pane .ui-layout-north { overflow: hidden; @@ -3246,7 +3190,7 @@ a:visited.ui-button { margin: 0 2px; } #ibo-main-content .ui-layout-button-pin-down, .ui-dialog .ui-layout-button-pin-down { - background: url("../../../../images/splitter-bkg.png?v=1605021905") transparent; + background: url("../../../../images/splitter-bkg.png?v=1605536038") transparent; width: 16px; background-position: -144px -144px; } @@ -3383,13 +3327,13 @@ a:visited.ui-button { #ibo-main-content .caselog_header, .ui-dialog .caselog_header { padding: 3px; border-top: 1px solid #fff; - background: #ddd url("../../../../images/plus.gif?v=1605021905") left no-repeat; + background: #ddd url("../../../../images/plus.gif?v=1605536038") left no-repeat; padding-left: 16px; cursor: pointer; width: 100%; } #ibo-main-content .caselog_header.open, .ui-dialog .caselog_header.open { - background: #ddd url("../../../../images/minus.gif?v=1605021905") left no-repeat; + background: #ddd url("../../../../images/minus.gif?v=1605536038") left no-repeat; } #ibo-main-content .caselog_entry, .ui-dialog .caselog_entry, #ibo-main-content .caselog_entry_html, .ui-dialog .caselog_entry_html { overflow-x: auto; @@ -3565,7 +3509,7 @@ a:visited.ui-button { height: 15px; border: 1px #A6A6A6 solid; cursor: pointer; - background-image: url("../../../../images/full-screen.png?v=1605021905"); + background-image: url("../../../../images/full-screen.png?v=1605536038"); background-repeat: no-repeat; background-position: center center; background-size: 98%; @@ -3612,7 +3556,7 @@ a:visited.ui-button { padding: 2px; } #ibo-main-content .field_container > div > div.field_value .attribute-edit .field_input_zone.field_input_document .button .ui-icon, .ui-dialog .field_container > div > div.field_value .attribute-edit .field_input_zone.field_input_document .button .ui-icon { - background-image: url("../../../../css/ui-lightness/images/ui-icons_ffffff_256x240.png?v=1605021905"); + background-image: url("../../../../css/ui-lightness/images/ui-icons_ffffff_256x240.png?v=1605536038"); background-color: #EA7D1E; } #ibo-main-content .field_container > div > div.field_value .attribute-edit .field_input_zone.field_input_image input, .ui-dialog .field_container > div > div.field_value .attribute-edit .field_input_zone.field_input_image input { @@ -3634,7 +3578,6 @@ a:visited.ui-button { width: 100%; } #ibo-main-content .field_container > div > div.field_value .attribute-edit .field_input_zone.field_input_extkey > .field_input_btn, .ui-dialog .field_container > div > div.field_value .attribute-edit .field_input_zone.field_input_extkey > .field_input_btn { - display: table-cell; width: 25px; padding-left: 0.4em; } @@ -3679,7 +3622,7 @@ a:visited.ui-button { height: 15px; border: 1px #A6A6A6 solid; cursor: pointer; - background-image: url("../../../../images/full-screen.png?v=1605021905"); + background-image: url("../../../../images/full-screen.png?v=1605536038"); background-repeat: no-repeat; background-position: center center; background-size: 98%; @@ -3744,7 +3687,7 @@ a:visited.ui-button { padding-left: 0.4em; } #ibo-main-content .ac_dlg_loading, .ui-dialog .ac_dlg_loading { - background: white url("../../../../images/indicator.gif?v=1605021905") right center no-repeat; + background: white url("../../../../images/indicator.gif?v=1605536038") right center no-repeat; } #ibo-main-content table.pagination, .ui-dialog table.pagination { display: inline-block; @@ -3811,10 +3754,10 @@ a:visited.ui-button { cursor: not-allowed; } #ibo-main-content .dragHover, .ui-dialog .dragHover { - background: url("../../../../css/ui-lightness/images/ui-bg_diagonals-thick_20_666666_40x40.png?v=1605021905"); + background: url("../../../../css/ui-lightness/images/ui-bg_diagonals-thick_20_666666_40x40.png?v=1605536038"); } #ibo-main-content .edit_mode .dashlet, .ui-dialog .edit_mode .dashlet { - background: url("../../../../css/ui-lightness/images/ui-bg_diagonals-thick_20_666666_40x40.png?v=1605021905"); + background: url("../../../../css/ui-lightness/images/ui-bg_diagonals-thick_20_666666_40x40.png?v=1605536038"); padding: 5px; margin: 0; position: relative; @@ -3859,7 +3802,7 @@ a:visited.ui-button { top: 0; right: 0; z-index: 10; - background: transparent url("../../../../images/delete.png?v=1605021905") no-repeat center; + background: transparent url("../../../../images/delete.png?v=1605536038") no-repeat center; } #ibo-main-content td.prop_value, .ui-dialog td.prop_value { text-align: left; @@ -4088,23 +4031,23 @@ a:visited.ui-button { } #ibo-main-content .message_info, .ui-dialog .message_info { border: 1px solid #993; - background: url("../../../../images/info-mini.png?v=1605021905") 1em 1em no-repeat #ffc; + background: url("../../../../images/info-mini.png?v=1605536038") 1em 1em no-repeat #ffc; padding-left: 3em; } #ibo-main-content .message_ok, .ui-dialog .message_ok { border: 1px solid #393; - background: url("../../../../images/ok.png?v=1605021905") 1em 1em no-repeat #cfc; + background: url("../../../../images/ok.png?v=1605536038") 1em 1em no-repeat #cfc; padding-left: 3em; } #ibo-main-content .message_warning, .ui-dialog .message_warning { border: 1px solid #ec9800; - background: url("../../../../images/error.png?v=1605021905") 1em 1em no-repeat #ffd78d; + background: url("../../../../images/error.png?v=1605536038") 1em 1em no-repeat #ffd78d; color: #000; padding-left: 3em; } #ibo-main-content .message_error, .ui-dialog .message_error { border: 1px solid #933; - background: url("../../../../images/error.png?v=1605021905") 1em 1em no-repeat #fcc; + background: url("../../../../images/error.png?v=1605536038") 1em 1em no-repeat #fcc; padding-left: 3em; } #ibo-main-content .fg-menu a img, .ui-dialog .fg-menu a img { @@ -4235,18 +4178,18 @@ a:visited.ui-button { } #ibo-main-content #hiddeable_chapters .ui-tabs .ui-tabs-nav li.hideable-chapter span, .ui-dialog #hiddeable_chapters .ui-tabs .ui-tabs-nav li.hideable-chapter span { padding-left: 20px; - background: url("../../../../images/eye-open-555.png?v=1605021905") 2px center no-repeat; + background: url("../../../../images/eye-open-555.png?v=1605536038") 2px center no-repeat; } #ibo-main-content #hiddeable_chapters .ui-tabs .ui-tabs-nav li.hideable-chapter.strikethrough span, .ui-dialog #hiddeable_chapters .ui-tabs .ui-tabs-nav li.hideable-chapter.strikethrough span { text-decoration: line-through; - background: url("../../../../images/eye-closed-555.png?v=1605021905") 2px center no-repeat; + background: url("../../../../images/eye-closed-555.png?v=1605536038") 2px center no-repeat; } #ibo-main-content .printable-version legend, .ui-dialog .printable-version legend { padding-left: 26px; - background: #1c94c4 url("../../../../images/eye-open-fff.png?v=1605021905") 8px center no-repeat; + background: #1c94c4 url("../../../../images/eye-open-fff.png?v=1605536038") 8px center no-repeat; } #ibo-main-content .printable-version .strikethrough legend, .ui-dialog .printable-version .strikethrough legend { - background: #1c94c4 url("../../../../images/eye-closed-fff.png?v=1605021905") 8px center no-repeat; + background: #1c94c4 url("../../../../images/eye-closed-fff.png?v=1605536038") 8px center no-repeat; } #ibo-main-content .printable-version fieldset.strikethrough span, .ui-dialog .printable-version fieldset.strikethrough span { display: none; @@ -4319,7 +4262,7 @@ a:visited.ui-button { cursor: pointer; width: 16px; height: 16px; - background-image: url("../../../../css/ui-lightness/images/ui-icons_222222_256x240.png?v=1605021905"); + background-image: url("../../../../css/ui-lightness/images/ui-icons_222222_256x240.png?v=1605536038"); background-position: -16px -192px; } #ibo-main-content .history_entry_truncated .history_truncated_toggler, .ui-dialog .history_entry_truncated .history_truncated_toggler { @@ -4404,7 +4347,7 @@ a:visited.ui-button { #ibo-main-content #itop-breadcrumb .breadcrumb-item a::after, .ui-dialog #itop-breadcrumb .breadcrumb-item a::after { content: ''; position: absolute; - background-image: url("../../../../images/breadcrumb-separator.png?v=1605021905"); + background-image: url("../../../../images/breadcrumb-separator.png?v=1605536038"); background-repeat: no-repeat; width: 8px; height: 16px; diff --git a/datamodels/2.x/itop-structure/precompiled-themes/test-red/main.css b/datamodels/2.x/itop-structure/precompiled-themes/test-red/main.css index 191e5425e..4cec20acf 100644 --- a/datamodels/2.x/itop-structure/precompiled-themes/test-red/main.css +++ b/datamodels/2.x/itop-structure/precompiled-themes/test-red/main.css @@ -1,6 +1,6 @@ /* === SIGNATURE BEGIN === -{"variables":"8cfe86f2c55d8eff36d57eb4e83d89f1","stylesheets":{"css-variables":"732f501302c538f594fb3c668d85b46c","jqueryui":"a887ad1c00f29132beba3bf729b981c7","main":"b04966c35dbdc85a2cd96abb6cd5fd6a","environment-banner":"3de3ffb8232b9a649e912b570a64bf5d"},"imports":[],"images":{"css\/ui-lightness\/images\/ui-icons_222222_256x240.png":"3a3c5468f484f07ac4a320d9e22acb8c","css\/ui-lightness\/images\/ui-bg_diagonals-thick_20_666666_40x40.png":"4429d568c67d8dfeb9040273ea0fb8c4","css\/ui-lightness\/images\/ui-icons_E87C1E_256x240.png":"7003dd36cb2aa032c8ec871ce4d4e03d","css\/ui-lightness\/images\/ui-icons_1c94c4_256x240.png":"dbd693dc8e0ef04e90a2f7ac7b390086","css\/ui-lightness\/images\/ui-icons_F26522_256x240.png":"16278ec0c07270be571f4c2e97fcc10c","css\/ui-lightness\/images\/ui-bg_diagonals-thick_18_b81900_40x40.png":"e460a66d4b3e093fc651e62a236267cb","css\/ui-lightness\/images\/ui-icons_ffffff_256x240.png":"41612b0f4a034424f8321c9f824a94da","css\/ui-lightness\/images\/ui-icons_ffd27a_256x240.png":"dda1b6f694b0d196aefc66a1d6d758f6","images\/actions_right.png":"31c8906bd25d27b83a0a2466bf903462","images\/ac-background.gif":"76135f3697b41a15aed787cfd77776c7","images\/green-square.gif":"16ea9a497d72f5e66e4e8ea9ae08024e","images\/tv-item.gif":"719fe2d4566108e73162fb8868d3778c","images\/tv-collapsable.gif":"63a3351ea0d580797c9b8c386aa4f48b","images\/tv-expandable.gif":"a2d1af4128e4a798a7f3390b12a28574","images\/tv-item-last.gif":"2ae7e1d9972ce71e5caa65a086bc5b7e","images\/tv-collapsable-last.gif":"71acaa9d7c2616e9e8b7131a75ca65da","images\/tv-expandable-last.gif":"9d51036b3a8102742709da66789fd0f7","images\/red-header.gif":"c73b8765f0c8c3c183cb6a0c2bb0ec69","images\/green-header.gif":"0e22a09bb8051b2a274b3427ede62e82","images\/orange-header.gif":"ce1f93f0af64431771b4cbd6c99c567b","images\/calendar.png":"ab56e59af3c96ca661821257d376465e","images\/truncated.png":"c6f91108afe8159d417b4dc556cd3b2a","images\/plus.gif":"f00e1e6e1161f48608bb2bbc79b9948c","images\/minus.gif":"6d77c0c0c2f86b6995d1cdf78274eaab","images\/full-screen.png":"b541fadd3f1563856a4b44aeebd9d563","images\/indicator.gif":"03ce3dcc84af110e9da8699a841e5200","images\/delete.png":"93c047549c31a270a037840277cf59d3","images\/info-mini.png":"445c090ed777c5e6a08ac390fa896193","images\/ok.png":"f6973773335fd83d8d2875f9a3c925af","images\/error.png":"1af8a1041016f67669c5fd22dc88c82e","images\/eye-open-555.png":"9940f4e5b1248042c238e1924359fd5e","images\/eye-closed-555.png":"6ad3b0bae791bf61addc9d8ca80a642d","images\/eye-open-fff.png":"b7db2402d4d5c72314c25790a66150d4","images\/eye-closed-fff.png":"f9be7454dbb47b0e0bca3aa370ae7db5"}} +{"variables":"8cfe86f2c55d8eff36d57eb4e83d89f1","stylesheets":{"css-variables":"732f501302c538f594fb3c668d85b46c","jqueryui":"a887ad1c00f29132beba3bf729b981c7","main":"d5d2002c5bcf8fcf592d8e028bbed2fb","environment-banner":"3de3ffb8232b9a649e912b570a64bf5d"},"imports":[],"images":{"css\/ui-lightness\/images\/ui-icons_222222_256x240.png":"3a3c5468f484f07ac4a320d9e22acb8c","css\/ui-lightness\/images\/ui-bg_diagonals-thick_20_666666_40x40.png":"4429d568c67d8dfeb9040273ea0fb8c4","css\/ui-lightness\/images\/ui-icons_E87C1E_256x240.png":"7003dd36cb2aa032c8ec871ce4d4e03d","css\/ui-lightness\/images\/ui-icons_1c94c4_256x240.png":"dbd693dc8e0ef04e90a2f7ac7b390086","css\/ui-lightness\/images\/ui-icons_F26522_256x240.png":"16278ec0c07270be571f4c2e97fcc10c","css\/ui-lightness\/images\/ui-bg_diagonals-thick_18_b81900_40x40.png":"e460a66d4b3e093fc651e62a236267cb","css\/ui-lightness\/images\/ui-icons_ffffff_256x240.png":"41612b0f4a034424f8321c9f824a94da","css\/ui-lightness\/images\/ui-icons_ffd27a_256x240.png":"dda1b6f694b0d196aefc66a1d6d758f6","images\/actions_right.png":"31c8906bd25d27b83a0a2466bf903462","images\/ac-background.gif":"76135f3697b41a15aed787cfd77776c7","images\/green-square.gif":"16ea9a497d72f5e66e4e8ea9ae08024e","images\/tv-item.gif":"719fe2d4566108e73162fb8868d3778c","images\/tv-collapsable.gif":"63a3351ea0d580797c9b8c386aa4f48b","images\/tv-expandable.gif":"a2d1af4128e4a798a7f3390b12a28574","images\/tv-item-last.gif":"2ae7e1d9972ce71e5caa65a086bc5b7e","images\/tv-collapsable-last.gif":"71acaa9d7c2616e9e8b7131a75ca65da","images\/tv-expandable-last.gif":"9d51036b3a8102742709da66789fd0f7","images\/red-header.gif":"c73b8765f0c8c3c183cb6a0c2bb0ec69","images\/green-header.gif":"0e22a09bb8051b2a274b3427ede62e82","images\/orange-header.gif":"ce1f93f0af64431771b4cbd6c99c567b","images\/calendar.png":"ab56e59af3c96ca661821257d376465e","images\/truncated.png":"c6f91108afe8159d417b4dc556cd3b2a","images\/plus.gif":"f00e1e6e1161f48608bb2bbc79b9948c","images\/minus.gif":"6d77c0c0c2f86b6995d1cdf78274eaab","images\/full-screen.png":"b541fadd3f1563856a4b44aeebd9d563","images\/indicator.gif":"03ce3dcc84af110e9da8699a841e5200","images\/delete.png":"93c047549c31a270a037840277cf59d3","images\/info-mini.png":"445c090ed777c5e6a08ac390fa896193","images\/ok.png":"f6973773335fd83d8d2875f9a3c925af","images\/error.png":"1af8a1041016f67669c5fd22dc88c82e","images\/eye-open-555.png":"9940f4e5b1248042c238e1924359fd5e","images\/eye-closed-555.png":"6ad3b0bae791bf61addc9d8ca80a642d","images\/eye-open-fff.png":"b7db2402d4d5c72314c25790a66150d4","images\/eye-closed-fff.png":"f9be7454dbb47b0e0bca3aa370ae7db5"}} === SIGNATURE END === */ /*! @@ -99,7 +99,7 @@ background-repeat: no-repeat; width: 16px; height: 16px; - background-image: url("../../../../css/ui-lightness/images/ui-icons_222222_256x240.png?v=1605021905"); + background-image: url("../../../../css/ui-lightness/images/ui-icons_222222_256x240.png?v=1605536038"); filter: hue-rotate(0deg); } .ui-widget-icon-block { @@ -232,7 +232,7 @@ .ui-menu .ui-menu-item { margin: 0; cursor: pointer; - list-style-image: url("?v=1605021905"); + list-style-image: url("?v=1605536038"); } .ui-menu .ui-menu-item-wrapper { position: relative; @@ -298,7 +298,7 @@ color: #EA7D1E; } .ui-button:hover .ui-icon { - background-image: url("../../../../css/ui-lightness/images/ui-icons_E87C1E_256x240.png?v=1605021905"); + background-image: url("../../../../css/ui-lightness/images/ui-icons_E87C1E_256x240.png?v=1605536038"); } .ui-button:active { text-decoration: none; @@ -306,20 +306,20 @@ color: #EA7D1E; } .ui-button:active .ui-icon { - background-image: url("../../../../css/ui-lightness/images/ui-icons_E87C1E_256x240.png?v=1605021905"); + background-image: url("../../../../css/ui-lightness/images/ui-icons_E87C1E_256x240.png?v=1605536038"); } .ui-button:focus { font-weight: bold; color: #EA7D1E; } .ui-button:focus .ui-icon { - background-image: url("../../../../css/ui-lightness/images/ui-icons_E87C1E_256x240.png?v=1605021905"); + background-image: url("../../../../css/ui-lightness/images/ui-icons_E87C1E_256x240.png?v=1605536038"); } .ui-button .ui-state-highlight.ui-icon { - background-image: url("../../../../css/ui-lightness/images/ui-icons_1c94c4_256x240.png?v=1605021905"); + background-image: url("../../../../css/ui-lightness/images/ui-icons_1c94c4_256x240.png?v=1605536038"); } .ui-button .ui-icon { - background-image: url("../../../../css/ui-lightness/images/ui-icons_F26522_256x240.png?v=1605021905"); + background-image: url("../../../../css/ui-lightness/images/ui-icons_F26522_256x240.png?v=1605536038"); } .ui-button-icon-only { width: 2em; @@ -901,7 +901,7 @@ body .ui-tooltip { background-image: none; } .ui-widget-content .ui-icon { - background-image: url("../../../../css/ui-lightness/images/ui-icons_222222_256x240.png?v=1605021905"); + background-image: url("../../../../css/ui-lightness/images/ui-icons_222222_256x240.png?v=1605536038"); } .ui-widget-header .ui-priority-secondary { opacity: 0.7; @@ -913,7 +913,7 @@ body .ui-tooltip { background-image: none; } .ui-widget-header .ui-icon { - background-image: url("../../../../css/ui-lightness/images/ui-icons_ffffff_256x240.png?v=1605021905"); + background-image: url("../../../../css/ui-lightness/images/ui-icons_ffffff_256x240.png?v=1605536038"); } .ui-state-default a { text-decoration: none; @@ -925,7 +925,7 @@ body .ui-tooltip { text-decoration: none; } .ui-state-default .ui-icon { - background-image: url("../../../../css/ui-lightness/images/ui-icons_F26522_256x240.png?v=1605021905"); + background-image: url("../../../../css/ui-lightness/images/ui-icons_F26522_256x240.png?v=1605536038"); } a.ui-button { text-decoration: none; @@ -955,7 +955,7 @@ a:visited.ui-button { text-decoration: none; } .ui-state-hover .ui-icon { - background-image: url("../../../../css/ui-lightness/images/ui-icons_E87C1E_256x240.png?v=1605021905"); + background-image: url("../../../../css/ui-lightness/images/ui-icons_E87C1E_256x240.png?v=1605536038"); } .ui-state-focus a { text-decoration: none; @@ -970,7 +970,7 @@ a:visited.ui-button { text-decoration: none; } .ui-state-focus .ui-icon { - background-image: url("../../../../css/ui-lightness/images/ui-icons_E87C1E_256x240.png?v=1605021905"); + background-image: url("../../../../css/ui-lightness/images/ui-icons_E87C1E_256x240.png?v=1605536038"); } .ui-visual-focus { box-shadow: 0 0 3px 1px #5e9ed6; @@ -988,19 +988,19 @@ a:visited.ui-button { text-decoration: none; } .ui-state-active .ui-icon { - background-image: url("../../../../css/ui-lightness/images/ui-icons_E87C1E_256x240.png?v=1605021905"); + background-image: url("../../../../css/ui-lightness/images/ui-icons_E87C1E_256x240.png?v=1605536038"); } .ui-icon-background { background-color: #EA7D1E; } .ui-state-highlight .ui-icon { - background-image: url("../../../../css/ui-lightness/images/ui-icons_1c94c4_256x240.png?v=1605021905"); + background-image: url("../../../../css/ui-lightness/images/ui-icons_1c94c4_256x240.png?v=1605536038"); } .ui-state-error .ui-icon { - background-image: url("../../../../css/ui-lightness/images/ui-icons_ffd27a_256x240.png?v=1605021905"); + background-image: url("../../../../css/ui-lightness/images/ui-icons_ffd27a_256x240.png?v=1605536038"); } .ui-state-error-text .ui-icon { - background-image: url("../../../../css/ui-lightness/images/ui-icons_ffd27a_256x240.png?v=1605021905"); + background-image: url("../../../../css/ui-lightness/images/ui-icons_ffd27a_256x240.png?v=1605536038"); } .ui-priority-primary { font-weight: bold; @@ -1643,52 +1643,10 @@ a:visited.ui-button { color: #EA7D1E; text-decoration: none; } -#ibo-main-content table.datatable, .ui-dialog table.datatable { - width: 100%; - border: 0; - padding: 0; -} #ibo-main-content td.menucontainer, .ui-dialog td.menucontainer { text-align: right; -} -#ibo-main-content table.listResults, .ui-dialog table.listResults { - padding: 0px; - border-top: 3px solid #F1F1F1; - border-left: 3px solid #F1F1F1; - border-bottom: 3px solid #e6e6e1; - border-right: 3px solid #e6e6e1; - width: 100%; - background-color: #FFFFFF; -} -#ibo-main-content table.listResults td, .ui-dialog table.listResults td { - padding: 2px; -} -#ibo-main-content table.listResults td .view-image, .ui-dialog table.listResults td .view-image { - width: inherit !important; - height: inherit !important; -} -#ibo-main-content table.listResults td .view-image img, .ui-dialog table.listResults td .view-image img { - max-width: 48px !important; - max-height: 48px !important; - display: block; - margin-left: auto; - margin-right: auto; -} -#ibo-main-content table.listResults > tbody > tr > *, .ui-dialog table.listResults > tbody > tr > * { - transition: background-color 400ms linear; -} -#ibo-main-content table.listResults > tbody > tr:hover > *, .ui-dialog table.listResults > tbody > tr:hover > * { - cursor: pointer; -} -#ibo-main-content table.listResults > tbody > tr.selected:hover > *, .ui-dialog table.listResults > tbody > tr.selected:hover > * { /* hover on lines is currently done toggling td.hover, and having a rule for links */ - background-color: #f1a564; - color: #000; -} -#ibo-main-content table.listResults > tbody > tr:hover > *, .ui-dialog table.listResults > tbody > tr:hover > * { /* hover on lines is currently done toggling td.hover, and having a rule for links */ - background-color: #fdf5d0; - color: #000; } #ibo-main-content .edit-image .view-image, .ui-dialog .edit-image .view-image { display: inline-block; @@ -1697,7 +1655,7 @@ a:visited.ui-button { visibility: hidden; } #ibo-main-content .edit-image .view-image.dirty.compat, .ui-dialog .edit-image .view-image.dirty.compat { - background-image: url("../../../../css/ui-lightness/images/ui-bg_diagonals-thick_20_666666_40x40.png?v=1605021905"); + background-image: url("../../../../css/ui-lightness/images/ui-bg_diagonals-thick_20_666666_40x40.png?v=1605536038"); } #ibo-main-content .edit-image .view-image.dirty.compat img, .ui-dialog .edit-image .view-image.dirty.compat img { opacity: 0.3; @@ -1720,7 +1678,7 @@ a:visited.ui-button { opacity: 0.3; } #ibo-main-content .edit-image .edit-buttons .button .ui-icon, .ui-dialog .edit-image .edit-buttons .button .ui-icon { - background-image: url("../../../../css/ui-lightness/images/ui-icons_ffffff_256x240.png?v=1605021905"); + background-image: url("../../../../css/ui-lightness/images/ui-icons_ffffff_256x240.png?v=1605536038"); } #ibo-main-content .edit-image .file-input, .ui-dialog .edit-image .file-input { display: block; @@ -1781,13 +1739,6 @@ a:visited.ui-button { #ibo-main-content tr td.hover, .ui-dialog tr td.hover, #ibo-main-content tr.even td.hover, .ui-dialog tr.even td.hover, #ibo-main-content .hover a, .ui-dialog .hover a, #ibo-main-content .hover a:visited, .ui-dialog .hover a:visited, #ibo-main-content .hover a:hover, .ui-dialog .hover a:hover, #ibo-main-content .wizContainer tr.even td.hover, .ui-dialog .wizContainer tr.even td.hover, #ibo-main-content .wizContainer tr td.hover, .ui-dialog .wizContainer tr td.hover { color: #000; } -#ibo-main-content th, .ui-dialog th { - font-family: Tahoma, Verdana, Arial, Helvetica; - font-size: 8pt; - color: #1c94c4; - height: 20px; - background: #F1F1F1 bottom repeat-x; -} #ibo-main-content th.header, .ui-dialog th.header { cursor: pointer; background-repeat: no-repeat; @@ -1940,7 +1891,7 @@ a:visited.ui-button { background: #EA7D1E; } #ibo-main-content .actions_details span, .ui-dialog .actions_details span { - background: url("../../../../images/actions_right.png?v=1605021905") no-repeat right; + background: url("../../../../images/actions_right.png?v=1605536038") no-repeat right; color: #fff; font-weight: bold; padding-top: 2px; @@ -1963,7 +1914,7 @@ a:visited.ui-button { } #ibo-main-content .ac_input, .ui-dialog .ac_input { border: 1px solid #7f9db9; - background: #fff url("../../../../images/ac-background.gif?v=1605021905") no-repeat right; + background: #fff url("../../../../images/ac-background.gif?v=1605536038") no-repeat right; /* By Rom */ } #ibo-main-content .csvimport_createobj, .ui-dialog .csvimport_createobj { @@ -2018,7 +1969,7 @@ a:visited.ui-button { noborder-top: 1px solid #8b8b8b; padding: 4px 0px 0px 16px; font-size: 8pt; - background: url("../../../../images/green-square.gif?v=1605021905") no-repeat bottom left; + background: url("../../../../images/green-square.gif?v=1605536038") no-repeat bottom left; color: #83b217; font-weight: bold; text-decoration: none; @@ -2094,22 +2045,22 @@ a:visited.ui-button { margin-top: -8px; } #ibo-main-content .notreeview li, .ui-dialog .notreeview li { - background: url("../../../../images/tv-item.gif?v=1605021905") 0 0 no-repeat; + background: url("../../../../images/tv-item.gif?v=1605536038") 0 0 no-repeat; } #ibo-main-content .notreeview .collapsable, .ui-dialog .notreeview .collapsable { - background-image: url("../../../../images/tv-collapsable.gif?v=1605021905"); + background-image: url("../../../../images/tv-collapsable.gif?v=1605536038"); } #ibo-main-content .notreeview .expandable, .ui-dialog .notreeview .expandable { - background-image: url("../../../../images/tv-expandable.gif?v=1605021905"); + background-image: url("../../../../images/tv-expandable.gif?v=1605536038"); } #ibo-main-content .notreeview .last, .ui-dialog .notreeview .last { - background-image: url("../../../../images/tv-item-last.gif?v=1605021905"); + background-image: url("../../../../images/tv-item-last.gif?v=1605536038"); } #ibo-main-content .notreeview .lastCollapsable, .ui-dialog .notreeview .lastCollapsable { - background-image: url("../../../../images/tv-collapsable-last.gif?v=1605021905"); + background-image: url("../../../../images/tv-collapsable-last.gif?v=1605536038"); } #ibo-main-content .notreeview .lastExpandable, .ui-dialog .notreeview .lastExpandable { - background-image: url("../../../../images/tv-expandable-last.gif?v=1605021905"); + background-image: url("../../../../images/tv-expandable-last.gif?v=1605536038"); } #ibo-main-content #OrganizationSelection, .ui-dialog #OrganizationSelection { padding: 5px 0px 16px 20px; @@ -2285,7 +2236,7 @@ a:visited.ui-button { color: #000; } #ibo-main-content th.red, .ui-dialog th.red { - background: url("../../../../images/red-header.gif?v=1605021905") bottom left repeat-x; + background: url("../../../../images/red-header.gif?v=1605536038") bottom left repeat-x; color: #000; } #ibo-main-content .green, .ui-dialog .green { @@ -2293,7 +2244,7 @@ a:visited.ui-button { color: #000; } #ibo-main-content th.green, .ui-dialog th.green { - background: url("../../../../images/green-header.gif?v=1605021905") bottom left repeat-x; + background: url("../../../../images/green-header.gif?v=1605536038") bottom left repeat-x; color: #000; } #ibo-main-content .orange, .ui-dialog .orange { @@ -2301,7 +2252,7 @@ a:visited.ui-button { color: #000; } #ibo-main-content th.orange, .ui-dialog th.orange { - background: url("../../../../images/orange-header.gif?v=1605021905") bottom left repeat-x; + background: url("../../../../images/orange-header.gif?v=1605536038") bottom left repeat-x; color: #000; /* For Date Picker: Creates a little calendar icon * instead of a text link for "Choose date" @@ -2316,7 +2267,7 @@ a:visited.ui-button { display: block; text-indent: -2000px; overflow: hidden; - background: url("../../../../images/calendar.png?v=1605021905") no-repeat; + background: url("../../../../images/calendar.png?v=1605536038") no-repeat; } #ibo-main-content td a.dp-choose-date.dp-disabled, .ui-dialog td a.dp-choose-date.dp-disabled, #ibo-main-content a.dp-choose-date.dp-disabled, .ui-dialog a.dp-choose-date.dp-disabled { background-position: 0 -20px; @@ -2368,15 +2319,6 @@ a:visited.ui-button { box-sizing: border-box; } #ibo-main-content .search_form_handler, .ui-dialog .search_form_handler { - position: relative; - z-index: 10; - width: 100%; - margin-left: auto; - margin-right: auto; - font-size: 12px; - text-align: left; - /* To compensate .search_box:text-align */ - border: 1px solid #1c94c4; /* Sizing reset */ } #ibo-main-content .search_form_handler *, .ui-dialog .search_form_handler * { @@ -2475,9 +2417,6 @@ a:visited.ui-button { /* padding-bottom must equals to padding-top - .search_form_criteria:margin-bottom */ background-color: #FFFFFF; } -#ibo-main-content .search_form_handler .sf_criterion_area .sf_criterion_row, .ui-dialog .search_form_handler .sf_criterion_area .sf_criterion_row { - position: relative; -} #ibo-main-content .search_form_handler .sf_criterion_area .sf_criterion_row:not(:first-child), .ui-dialog .search_form_handler .sf_criterion_area .sf_criterion_row:not(:first-child) { margin-top: 20px; } @@ -2503,6 +2442,11 @@ a:visited.ui-button { } #ibo-main-content .search_form_handler .sf_criterion_area .sf_criterion_row .sf_criterion_group, .ui-dialog .search_form_handler .sf_criterion_area .sf_criterion_row .sf_criterion_group { display: inline; +} +#ibo-main-content .search_form_handler .sf_criterion_area .sf_criterion_row .sf_criterion_group .sfc_fg_button, .ui-dialog .search_form_handler .sf_criterion_area .sf_criterion_row .sf_criterion_group .sfc_fg_button, #ibo-main-content .search_form_handler .sf_criterion_area .sf_criterion_row .sf_criterion_group .sfc_header, .ui-dialog .search_form_handler .sf_criterion_area .sf_criterion_row .sf_criterion_group .sfc_header { + border: 1px solid #E1E7EC; + /* Must be equal to .search_form_criteria:margin-bottom + this:padding-bottom */ + border-radius: 3px; /* Common style between criterion and more criterion */ } #ibo-main-content .search_form_handler .sf_criterion_area .search_form_criteria, .ui-dialog .search_form_handler .sf_criterion_area .search_form_criteria, #ibo-main-content .search_form_handler .sf_criterion_area .sf_more_criterion, .ui-dialog .search_form_handler .sf_criterion_area .sf_more_criterion, #ibo-main-content .search_form_handler .sf_criterion_area .sf_button, .ui-dialog .search_form_handler .sf_criterion_area .sf_button { @@ -2529,14 +2473,14 @@ a:visited.ui-button { border-radius: 1px; box-shadow: 0 1px 1px rgba(0, 0, 0, 0.15); } -#ibo-main-content .search_form_handler .sf_criterion_area .search_form_criteria .sfc_form_group, .ui-dialog .search_form_handler .sf_criterion_area .search_form_criteria .sfc_form_group, #ibo-main-content .search_form_handler .sf_criterion_area .sf_more_criterion .sfc_form_group, .ui-dialog .search_form_handler .sf_criterion_area .sf_more_criterion .sfc_form_group, #ibo-main-content .search_form_handler .sf_criterion_area .sf_button .sfc_form_group, .ui-dialog .search_form_handler .sf_criterion_area .sf_button .sfc_form_group, #ibo-main-content .search_form_handler .sf_criterion_area .search_form_criteria .sfm_content, .ui-dialog .search_form_handler .sf_criterion_area .search_form_criteria .sfm_content, #ibo-main-content .search_form_handler .sf_criterion_area .sf_more_criterion .sfm_content, .ui-dialog .search_form_handler .sf_criterion_area .sf_more_criterion .sfm_content, #ibo-main-content .search_form_handler .sf_criterion_area .sf_button .sfm_content, .ui-dialog .search_form_handler .sf_criterion_area .sf_button .sfm_content { +#ibo-main-content .search_form_handler .sf_criterion_area .search_form_criteria .sfm_content, .ui-dialog .search_form_handler .sf_criterion_area .search_form_criteria .sfm_content, #ibo-main-content .search_form_handler .sf_criterion_area .sf_more_criterion .sfm_content, .ui-dialog .search_form_handler .sf_criterion_area .sf_more_criterion .sfm_content, #ibo-main-content .search_form_handler .sf_criterion_area .sf_button .sfm_content, .ui-dialog .search_form_handler .sf_criterion_area .sf_button .sfm_content { position: absolute; z-index: -1; min-width: 100%; left: 0px; margin-top: -1px; } -#ibo-main-content .search_form_handler .sf_criterion_area .search_form_criteria .sfc_form_group .sfc_fg_buttons, .ui-dialog .search_form_handler .sf_criterion_area .search_form_criteria .sfc_form_group .sfc_fg_buttons, #ibo-main-content .search_form_handler .sf_criterion_area .sf_more_criterion .sfc_form_group .sfc_fg_buttons, .ui-dialog .search_form_handler .sf_criterion_area .sf_more_criterion .sfc_form_group .sfc_fg_buttons, #ibo-main-content .search_form_handler .sf_criterion_area .sf_button .sfc_form_group .sfc_fg_buttons, .ui-dialog .search_form_handler .sf_criterion_area .sf_button .sfc_form_group .sfc_fg_buttons, #ibo-main-content .search_form_handler .sf_criterion_area .search_form_criteria .sfm_content .sfc_fg_buttons, .ui-dialog .search_form_handler .sf_criterion_area .search_form_criteria .sfm_content .sfc_fg_buttons, #ibo-main-content .search_form_handler .sf_criterion_area .sf_more_criterion .sfm_content .sfc_fg_buttons, .ui-dialog .search_form_handler .sf_criterion_area .sf_more_criterion .sfm_content .sfc_fg_buttons, #ibo-main-content .search_form_handler .sf_criterion_area .sf_button .sfm_content .sfc_fg_buttons, .ui-dialog .search_form_handler .sf_criterion_area .sf_button .sfm_content .sfc_fg_buttons { +#ibo-main-content .search_form_handler .sf_criterion_area .search_form_criteria .sfm_content .sfc_fg_buttons, .ui-dialog .search_form_handler .sf_criterion_area .search_form_criteria .sfm_content .sfc_fg_buttons, #ibo-main-content .search_form_handler .sf_criterion_area .sf_more_criterion .sfm_content .sfc_fg_buttons, .ui-dialog .search_form_handler .sf_criterion_area .sf_more_criterion .sfm_content .sfc_fg_buttons, #ibo-main-content .search_form_handler .sf_criterion_area .sf_button .sfm_content .sfc_fg_buttons, .ui-dialog .search_form_handler .sf_criterion_area .sf_button .sfm_content .sfc_fg_buttons { white-space: nowrap; /* Criteria tags */ } @@ -2983,19 +2927,19 @@ a:visited.ui-button { /* Beware: IE6 does not support multiple selector with multiple classes, only the last class is used */ } #ibo-main-content table.listResults tr.odd td.truncated, .ui-dialog table.listResults tr.odd td.truncated, #ibo-main-content table.listResults tr td.truncated, .ui-dialog table.listResults tr td.truncated, #ibo-main-content .wizContainer table.listResults tr.odd td.truncated, .ui-dialog .wizContainer table.listResults tr.odd td.truncated, #ibo-main-content .wizContainer table.listResults tr td.truncated, .ui-dialog .wizContainer table.listResults tr td.truncated { - background: url("../../../../images/truncated.png?v=1605021905") bottom repeat-x; + background: url("../../../../images/truncated.png?v=1605536038") bottom repeat-x; /* Beware: IE6 does not support multiple selector with multiple classes, only the last class is used */ } #ibo-main-content table.listResults tr.even td.truncated, .ui-dialog table.listResults tr.even td.truncated, #ibo-main-content .wizContainer table.listResults tr.even td.truncated, .ui-dialog .wizContainer table.listResults tr.even td.truncated { - background: #f9f9f1 url("../../../../images/truncated.png?v=1605021905") bottom repeat-x; + background: #f9f9f1 url("../../../../images/truncated.png?v=1605536038") bottom repeat-x; /* Beware: IE6 does not support multiple selector with multiple classes, only the last class is used */ } #ibo-main-content table.listResults tr.even td.hover.truncated, .ui-dialog table.listResults tr.even td.hover.truncated, #ibo-main-content .wizContainer table.listResults tr.even td.hover.truncated, .ui-dialog .wizContainer table.listResults tr.even td.hover.truncated { - background: #fdf5d0 url("../../../../images/truncated.png?v=1605021905") bottom repeat-x; + background: #fdf5d0 url("../../../../images/truncated.png?v=1605536038") bottom repeat-x; /* Beware: IE6 does not support multiple selector with multiple classes, only the last class is used */ } #ibo-main-content table.listResults tr.odd td.hover.truncated, .ui-dialog table.listResults tr.odd td.hover.truncated, #ibo-main-content table.listResults tr td.hover.truncated, .ui-dialog table.listResults tr td.hover.truncated, #ibo-main-content .wizContainer table.listResults tr.odd td.hover.truncated, .ui-dialog .wizContainer table.listResults tr.odd td.hover.truncated, #ibo-main-content .wizContainer table.listResults tr td.hover.truncated, .ui-dialog .wizContainer table.listResults tr td.hover.truncated { - background: #fdf5d0 url("../../../../images/truncated.png?v=1605021905") bottom repeat-x; + background: #fdf5d0 url("../../../../images/truncated.png?v=1605536038") bottom repeat-x; } #ibo-main-content table.listResults.truncated, .ui-dialog table.listResults.truncated { border-bottom: 0; @@ -3097,7 +3041,7 @@ a:visited.ui-button { #ibo-main-content div#logo div, .ui-dialog div#logo div { height: 88px; width: 244px; - background: url("../../../../images/itop-logo-2.png?v=1605021905") left no-repeat; + background: url("../../../../images/itop-logo-2.png?v=1605536038") left no-repeat; } #ibo-main-content #left-pane .ui-layout-north, .ui-dialog #left-pane .ui-layout-north { overflow: hidden; @@ -3246,7 +3190,7 @@ a:visited.ui-button { margin: 0 2px; } #ibo-main-content .ui-layout-button-pin-down, .ui-dialog .ui-layout-button-pin-down { - background: url("../../../../images/splitter-bkg.png?v=1605021905") transparent; + background: url("../../../../images/splitter-bkg.png?v=1605536038") transparent; width: 16px; background-position: -144px -144px; } @@ -3383,13 +3327,13 @@ a:visited.ui-button { #ibo-main-content .caselog_header, .ui-dialog .caselog_header { padding: 3px; border-top: 1px solid #fff; - background: #ddd url("../../../../images/plus.gif?v=1605021905") left no-repeat; + background: #ddd url("../../../../images/plus.gif?v=1605536038") left no-repeat; padding-left: 16px; cursor: pointer; width: 100%; } #ibo-main-content .caselog_header.open, .ui-dialog .caselog_header.open { - background: #ddd url("../../../../images/minus.gif?v=1605021905") left no-repeat; + background: #ddd url("../../../../images/minus.gif?v=1605536038") left no-repeat; } #ibo-main-content .caselog_entry, .ui-dialog .caselog_entry, #ibo-main-content .caselog_entry_html, .ui-dialog .caselog_entry_html { overflow-x: auto; @@ -3565,7 +3509,7 @@ a:visited.ui-button { height: 15px; border: 1px #A6A6A6 solid; cursor: pointer; - background-image: url("../../../../images/full-screen.png?v=1605021905"); + background-image: url("../../../../images/full-screen.png?v=1605536038"); background-repeat: no-repeat; background-position: center center; background-size: 98%; @@ -3612,7 +3556,7 @@ a:visited.ui-button { padding: 2px; } #ibo-main-content .field_container > div > div.field_value .attribute-edit .field_input_zone.field_input_document .button .ui-icon, .ui-dialog .field_container > div > div.field_value .attribute-edit .field_input_zone.field_input_document .button .ui-icon { - background-image: url("../../../../css/ui-lightness/images/ui-icons_ffffff_256x240.png?v=1605021905"); + background-image: url("../../../../css/ui-lightness/images/ui-icons_ffffff_256x240.png?v=1605536038"); background-color: #EA7D1E; } #ibo-main-content .field_container > div > div.field_value .attribute-edit .field_input_zone.field_input_image input, .ui-dialog .field_container > div > div.field_value .attribute-edit .field_input_zone.field_input_image input { @@ -3634,7 +3578,6 @@ a:visited.ui-button { width: 100%; } #ibo-main-content .field_container > div > div.field_value .attribute-edit .field_input_zone.field_input_extkey > .field_input_btn, .ui-dialog .field_container > div > div.field_value .attribute-edit .field_input_zone.field_input_extkey > .field_input_btn { - display: table-cell; width: 25px; padding-left: 0.4em; } @@ -3679,7 +3622,7 @@ a:visited.ui-button { height: 15px; border: 1px #A6A6A6 solid; cursor: pointer; - background-image: url("../../../../images/full-screen.png?v=1605021905"); + background-image: url("../../../../images/full-screen.png?v=1605536038"); background-repeat: no-repeat; background-position: center center; background-size: 98%; @@ -3744,7 +3687,7 @@ a:visited.ui-button { padding-left: 0.4em; } #ibo-main-content .ac_dlg_loading, .ui-dialog .ac_dlg_loading { - background: white url("../../../../images/indicator.gif?v=1605021905") right center no-repeat; + background: white url("../../../../images/indicator.gif?v=1605536038") right center no-repeat; } #ibo-main-content table.pagination, .ui-dialog table.pagination { display: inline-block; @@ -3811,10 +3754,10 @@ a:visited.ui-button { cursor: not-allowed; } #ibo-main-content .dragHover, .ui-dialog .dragHover { - background: url("../../../../css/ui-lightness/images/ui-bg_diagonals-thick_20_666666_40x40.png?v=1605021905"); + background: url("../../../../css/ui-lightness/images/ui-bg_diagonals-thick_20_666666_40x40.png?v=1605536038"); } #ibo-main-content .edit_mode .dashlet, .ui-dialog .edit_mode .dashlet { - background: url("../../../../css/ui-lightness/images/ui-bg_diagonals-thick_20_666666_40x40.png?v=1605021905"); + background: url("../../../../css/ui-lightness/images/ui-bg_diagonals-thick_20_666666_40x40.png?v=1605536038"); padding: 5px; margin: 0; position: relative; @@ -3859,7 +3802,7 @@ a:visited.ui-button { top: 0; right: 0; z-index: 10; - background: transparent url("../../../../images/delete.png?v=1605021905") no-repeat center; + background: transparent url("../../../../images/delete.png?v=1605536038") no-repeat center; } #ibo-main-content td.prop_value, .ui-dialog td.prop_value { text-align: left; @@ -4088,23 +4031,23 @@ a:visited.ui-button { } #ibo-main-content .message_info, .ui-dialog .message_info { border: 1px solid #993; - background: url("../../../../images/info-mini.png?v=1605021905") 1em 1em no-repeat #ffc; + background: url("../../../../images/info-mini.png?v=1605536038") 1em 1em no-repeat #ffc; padding-left: 3em; } #ibo-main-content .message_ok, .ui-dialog .message_ok { border: 1px solid #393; - background: url("../../../../images/ok.png?v=1605021905") 1em 1em no-repeat #cfc; + background: url("../../../../images/ok.png?v=1605536038") 1em 1em no-repeat #cfc; padding-left: 3em; } #ibo-main-content .message_warning, .ui-dialog .message_warning { border: 1px solid #ec9800; - background: url("../../../../images/error.png?v=1605021905") 1em 1em no-repeat #ffd78d; + background: url("../../../../images/error.png?v=1605536038") 1em 1em no-repeat #ffd78d; color: #000; padding-left: 3em; } #ibo-main-content .message_error, .ui-dialog .message_error { border: 1px solid #933; - background: url("../../../../images/error.png?v=1605021905") 1em 1em no-repeat #fcc; + background: url("../../../../images/error.png?v=1605536038") 1em 1em no-repeat #fcc; padding-left: 3em; } #ibo-main-content .fg-menu a img, .ui-dialog .fg-menu a img { @@ -4235,18 +4178,18 @@ a:visited.ui-button { } #ibo-main-content #hiddeable_chapters .ui-tabs .ui-tabs-nav li.hideable-chapter span, .ui-dialog #hiddeable_chapters .ui-tabs .ui-tabs-nav li.hideable-chapter span { padding-left: 20px; - background: url("../../../../images/eye-open-555.png?v=1605021905") 2px center no-repeat; + background: url("../../../../images/eye-open-555.png?v=1605536038") 2px center no-repeat; } #ibo-main-content #hiddeable_chapters .ui-tabs .ui-tabs-nav li.hideable-chapter.strikethrough span, .ui-dialog #hiddeable_chapters .ui-tabs .ui-tabs-nav li.hideable-chapter.strikethrough span { text-decoration: line-through; - background: url("../../../../images/eye-closed-555.png?v=1605021905") 2px center no-repeat; + background: url("../../../../images/eye-closed-555.png?v=1605536038") 2px center no-repeat; } #ibo-main-content .printable-version legend, .ui-dialog .printable-version legend { padding-left: 26px; - background: #1c94c4 url("../../../../images/eye-open-fff.png?v=1605021905") 8px center no-repeat; + background: #1c94c4 url("../../../../images/eye-open-fff.png?v=1605536038") 8px center no-repeat; } #ibo-main-content .printable-version .strikethrough legend, .ui-dialog .printable-version .strikethrough legend { - background: #1c94c4 url("../../../../images/eye-closed-fff.png?v=1605021905") 8px center no-repeat; + background: #1c94c4 url("../../../../images/eye-closed-fff.png?v=1605536038") 8px center no-repeat; } #ibo-main-content .printable-version fieldset.strikethrough span, .ui-dialog .printable-version fieldset.strikethrough span { display: none; @@ -4319,7 +4262,7 @@ a:visited.ui-button { cursor: pointer; width: 16px; height: 16px; - background-image: url("../../../../css/ui-lightness/images/ui-icons_222222_256x240.png?v=1605021905"); + background-image: url("../../../../css/ui-lightness/images/ui-icons_222222_256x240.png?v=1605536038"); background-position: -16px -192px; } #ibo-main-content .history_entry_truncated .history_truncated_toggler, .ui-dialog .history_entry_truncated .history_truncated_toggler { @@ -4404,7 +4347,7 @@ a:visited.ui-button { #ibo-main-content #itop-breadcrumb .breadcrumb-item a::after, .ui-dialog #itop-breadcrumb .breadcrumb-item a::after { content: ''; position: absolute; - background-image: url("../../../../images/breadcrumb-separator.png?v=1605021905"); + background-image: url("../../../../images/breadcrumb-separator.png?v=1605536038"); background-repeat: no-repeat; width: 8px; height: 16px; diff --git a/dictionaries/cs.dictionary.itop.ui.php b/dictionaries/cs.dictionary.itop.ui.php index a876cb021..45946c3e7 100755 --- a/dictionaries/cs.dictionary.itop.ui.php +++ b/dictionaries/cs.dictionary.itop.ui.php @@ -1167,7 +1167,8 @@ Dict::Add('CS CZ', 'Czech', 'Čeština', array( 'UI:ColumnsAndSortOrder' => 'Sloupce a jejich řazení:', 'UI:UseDefaultSettings' => 'Použít výchozí nastavení', 'UI:UseSpecificSettings' => 'Použít následující nastavení:', - 'UI:Display_X_ItemsPerPage' => 'Zobrazit %1$s položek na stránku', + 'UI:Display_X_ItemsPerPage_prefix' => 'Zobrazit', + 'UI:Display_X_ItemsPerPage_suffix' => 'položek na stránku', 'UI:UseSavetheSettings' => 'Uložit nastavení', 'UI:OnlyForThisList' => 'Jen pro tento seznam', 'UI:ForAllLists' => 'Pro všechny seznamy', diff --git a/dictionaries/da.dictionary.itop.ui.php b/dictionaries/da.dictionary.itop.ui.php index a9580962e..ad527e31c 100644 --- a/dictionaries/da.dictionary.itop.ui.php +++ b/dictionaries/da.dictionary.itop.ui.php @@ -1156,7 +1156,8 @@ Ved tilknytningen til en trigger, bliver hver handling tildelt et "rækkefølge" 'UI:ColumnsAndSortOrder' => 'Kolonner og sortering:', 'UI:UseDefaultSettings' => 'Brug de anbefalede indstillinger', 'UI:UseSpecificSettings' => 'Brug følgende indstillinger:', - 'UI:Display_X_ItemsPerPage' => 'Vis %1$s emner per side', + 'UI:Display_X_ItemsPerPage_prefix' => 'Vis', + 'UI:Display_X_ItemsPerPage_suffix' => 'emner per side', 'UI:UseSavetheSettings' => 'Gem indstillinger', 'UI:OnlyForThisList' => 'Kun for denne liste', 'UI:ForAllLists' => 'For alle lister', diff --git a/dictionaries/de.dictionary.itop.ui.php b/dictionaries/de.dictionary.itop.ui.php index 65e9ef506..c4bccd924 100644 --- a/dictionaries/de.dictionary.itop.ui.php +++ b/dictionaries/de.dictionary.itop.ui.php @@ -1154,7 +1154,8 @@ Wenn Aktionen mit Trigger verknüpft sind, bekommt jede Aktion eine Auftragsnumm 'UI:ColumnsAndSortOrder' => 'Spalten und Sortierrheienfolge:', 'UI:UseDefaultSettings' => 'Verwende Default-Einstellungen', 'UI:UseSpecificSettings' => 'Verwende folgende Einstellungen:', - 'UI:Display_X_ItemsPerPage' => '%1$s Elemente pro Seite anzeigen', + 'UI:Display_X_ItemsPerPage_prefix' => '', + 'UI:Display_X_ItemsPerPage_suffix' => 'Elemente pro Seite anzeigen', 'UI:UseSavetheSettings' => 'Einstellungen speichern', 'UI:OnlyForThisList' => 'Nur für diese Liste', 'UI:ForAllLists' => 'Standard für alle Listen', diff --git a/dictionaries/en.dictionary.itop.ui.php b/dictionaries/en.dictionary.itop.ui.php index 05746117c..58c61c314 100644 --- a/dictionaries/en.dictionary.itop.ui.php +++ b/dictionaries/en.dictionary.itop.ui.php @@ -1173,7 +1173,8 @@ When associated with a trigger, each action is given an "order" number, specifyi 'UI:ColumnsAndSortOrder' => 'Columns and sort order:', 'UI:UseDefaultSettings' => 'Use the Default Settings', 'UI:UseSpecificSettings' => 'Use the Following Settings:', - 'UI:Display_X_ItemsPerPage' => 'Display %1$s items per page', + 'UI:Display_X_ItemsPerPage_prefix' => 'Display', + 'UI:Display_X_ItemsPerPage_suffix' => 'items per page', 'UI:UseSavetheSettings' => 'Save the Settings', 'UI:OnlyForThisList' => 'Only for this list', 'UI:ForAllLists' => 'Default for all lists', diff --git a/dictionaries/es_cr.dictionary.itop.ui.php b/dictionaries/es_cr.dictionary.itop.ui.php index 0e5699ad1..d5c1ab9c7 100644 --- a/dictionaries/es_cr.dictionary.itop.ui.php +++ b/dictionaries/es_cr.dictionary.itop.ui.php @@ -1169,7 +1169,8 @@ Cuando se asocien con un disparador, cada acción recibe un número de "orden", 'UI:ColumnsAndSortOrder' => 'Columnas y Ordenamiento:', 'UI:UseDefaultSettings' => 'Usar Configuración por Omisión', 'UI:UseSpecificSettings' => 'Usar la Siguiente Configuración:', - 'UI:Display_X_ItemsPerPage' => 'Desplegar %1$s elementos por página', + 'UI:Display_X_ItemsPerPage_prefix' => 'Desplegar', + 'UI:Display_X_ItemsPerPage_suffix' => 'elementos por página', 'UI:UseSavetheSettings' => 'Guardar Configuraciones', 'UI:OnlyForThisList' => 'Sólo esta Lista', 'UI:ForAllLists' => 'Defecto en todas las listas', diff --git a/dictionaries/fr.dictionary.itop.ui.php b/dictionaries/fr.dictionary.itop.ui.php index 296d3ab2e..6bd6b5e0d 100644 --- a/dictionaries/fr.dictionary.itop.ui.php +++ b/dictionaries/fr.dictionary.itop.ui.php @@ -1153,7 +1153,8 @@ Lors de l\'association à un déclencheur, on attribue à chaque action un numé 'UI:ColumnsAndSortOrder' => 'Colonnes et ordre de tri:', 'UI:UseDefaultSettings' => 'Utiliser les réglages par défaut', 'UI:UseSpecificSettings' => 'Utiliser les réglages suivants:', - 'UI:Display_X_ItemsPerPage' => 'Afficher %1$s éléments par page', + 'UI:Display_X_ItemsPerPage_prefix' => 'Afficher', + 'UI:Display_X_ItemsPerPage_suffix' => 'éléments par page', 'UI:UseSavetheSettings' => 'Enregistrer ces réglages', 'UI:OnlyForThisList' => 'Seulement pour cette liste', 'UI:ForAllLists' => 'Défaut pour toutes les listes', diff --git a/dictionaries/ja.dictionary.itop.ui.php b/dictionaries/ja.dictionary.itop.ui.php index 39644c3ab..a2b90f9c6 100644 --- a/dictionaries/ja.dictionary.itop.ui.php +++ b/dictionaries/ja.dictionary.itop.ui.php @@ -1154,7 +1154,8 @@ Dict::Add('JA JP', 'Japanese', '日本語', array( 'UI:ColumnsAndSortOrder' => 'カラムと並び順:', 'UI:UseDefaultSettings' => '既定のセッティングを使用', 'UI:UseSpecificSettings' => '次のセッティングを使用:', - 'UI:Display_X_ItemsPerPage' => '1ページに %1$s アイテムを表示', + 'UI:Display_X_ItemsPerPage_prefix' => '1ページに', + 'UI:Display_X_ItemsPerPage_suffix' => 'アイテムを表示', 'UI:UseSavetheSettings' => 'セッティングを保存', 'UI:OnlyForThisList' => 'このリストのみ', 'UI:ForAllLists' => 'すべてのリストのデフォルト', diff --git a/dictionaries/nl.dictionary.itop.ui.php b/dictionaries/nl.dictionary.itop.ui.php index 0991e5bf0..f134d8442 100644 --- a/dictionaries/nl.dictionary.itop.ui.php +++ b/dictionaries/nl.dictionary.itop.ui.php @@ -1177,7 +1177,8 @@ Bij die koppeling wordt aan elke actie een volgorde-nummer gegeven. Dit bepaalt 'UI:ColumnsAndSortOrder' => 'Kolommen en sorteervolgorde:', 'UI:UseDefaultSettings' => 'Gebruik de standaard instellingen', 'UI:UseSpecificSettings' => 'Gebruik de volgende instellingen:', - 'UI:Display_X_ItemsPerPage' => 'Geef %1$s items per pagina weer', + 'UI:Display_X_ItemsPerPage_prefix' => 'Geef', + 'UI:Display_X_ItemsPerPage_suffix' => 'items per pagina weer', 'UI:UseSavetheSettings' => 'Sla de instellingen op', 'UI:OnlyForThisList' => 'Alleen voor deze lijst', 'UI:ForAllLists' => 'Standaard voor alle lijsten', diff --git a/dictionaries/pt_br.dictionary.itop.ui.php b/dictionaries/pt_br.dictionary.itop.ui.php index f435f7944..2d03baa08 100644 --- a/dictionaries/pt_br.dictionary.itop.ui.php +++ b/dictionaries/pt_br.dictionary.itop.ui.php @@ -1167,7 +1167,8 @@ When associated with a trigger, each action is given an "order" number, specifyi 'UI:ColumnsAndSortOrder' => 'Colunas e ordem de classificação:', 'UI:UseDefaultSettings' => 'Use a configuração padrão', 'UI:UseSpecificSettings' => 'Use as seguintes configurações:', - 'UI:Display_X_ItemsPerPage' => 'Mostrar %1$s itens por página', + 'UI:Display_X_ItemsPerPage_prefix' => 'Mostrar', + 'UI:Display_X_ItemsPerPage_suffix' => 'itens por página', 'UI:UseSavetheSettings' => 'Salvar configurações', 'UI:OnlyForThisList' => 'Somente para esta lista', 'UI:ForAllLists' => 'Para todas as listas', diff --git a/dictionaries/ru.dictionary.itop.ui.php b/dictionaries/ru.dictionary.itop.ui.php index c95697a79..a0d6ba9ff 100644 --- a/dictionaries/ru.dictionary.itop.ui.php +++ b/dictionaries/ru.dictionary.itop.ui.php @@ -1145,7 +1145,8 @@ Dict::Add('RU RU', 'Russian', 'Русский', array( 'UI:ColumnsAndSortOrder' => 'Колонки и порядок сортировки:', 'UI:UseDefaultSettings' => 'Использовать настройки по умолчанию', 'UI:UseSpecificSettings' => 'Использовать эти настройки:', - 'UI:Display_X_ItemsPerPage' => 'Показывать %1$s элементов на странице', + 'UI:Display_X_ItemsPerPage_prefix' => 'Показывать', + 'UI:Display_X_ItemsPerPage_suffix' => 'элементов на странице', 'UI:UseSavetheSettings' => 'Сохранить настройки', 'UI:OnlyForThisList' => 'Только для текущего списка', 'UI:ForAllLists' => 'Для всех списков', diff --git a/dictionaries/sk.dictionary.itop.ui.php b/dictionaries/sk.dictionary.itop.ui.php index d2b164b05..8505aeac6 100644 --- a/dictionaries/sk.dictionary.itop.ui.php +++ b/dictionaries/sk.dictionary.itop.ui.php @@ -1156,7 +1156,8 @@ Keď sú priradené spúštačom, každej akcii je dané číslo "príkazu", šp 'UI:ColumnsAndSortOrder' => 'Stĺpce a triediaci príkaz:', 'UI:UseDefaultSettings' => 'Použite štandardné nastavenia', 'UI:UseSpecificSettings' => 'Použite nasledovné nastavenia:', - 'UI:Display_X_ItemsPerPage' => 'Zobraziť %1$s položiek na stránku', + 'UI:Display_X_ItemsPerPage_prefix' => 'Zobraziť', + 'UI:Display_X_ItemsPerPage_suffix' => 'položiek na stránku', 'UI:UseSavetheSettings' => 'Uložiť nastavenia', 'UI:OnlyForThisList' => 'Iba pre tento zoznam', 'UI:ForAllLists' => 'Pre všetky zoznamy', diff --git a/dictionaries/ui/components/en.dictionary.itop.datatable.php b/dictionaries/ui/components/en.dictionary.itop.datatable.php new file mode 100644 index 000000000..281bed56f --- /dev/null +++ b/dictionaries/ui/components/en.dictionary.itop.datatable.php @@ -0,0 +1,33 @@ + 'Please wait...', + 'UI:Datatables:Language:Search' => 'Filter:', + 'UI:Datatables:Language:LengthMenu' => 'Showing _MENU_ out of', + 'UI:Datatables:Language:ZeroRecords' => 'No result', + 'UI:Datatables:Language:Info' => '_TOTAL_ items', + 'UI:Datatables:Language:InfoEmpty' => 'No information', + 'UI:Datatables:Language:InfoFiltered' => 'filtered out of _MAX_ items', + 'UI:Datatables:Language:EmptyTable' => 'No data available in this table', + 'UI:Datatables:Language:DisplayLength:All' => 'All', + 'UI:Datatables:Language:Sort:Ascending' => 'enable for an ascending sort', + 'UI:Datatables:Language:Sort:Descending' => 'enable for a descending sort', +)); \ No newline at end of file diff --git a/dictionaries/zh_cn.dictionary.itop.ui.php b/dictionaries/zh_cn.dictionary.itop.ui.php index 52bca2ff4..32335cd68 100644 --- a/dictionaries/zh_cn.dictionary.itop.ui.php +++ b/dictionaries/zh_cn.dictionary.itop.ui.php @@ -1167,7 +1167,8 @@ Dict::Add('ZH CN', 'Chinese', '简体中文', array( 'UI:ColumnsAndSortOrder' => '列和排序顺序:', 'UI:UseDefaultSettings' => '使用默认配置', 'UI:UseSpecificSettings' => '使用下面的配置:', - 'UI:Display_X_ItemsPerPage' => '每页显示 %1$s 个项目', + 'UI:Display_X_ItemsPerPage_prefix' => '每页显示', + 'UI:Display_X_ItemsPerPage_suffix' => '个项目', 'UI:UseSavetheSettings' => '保存设置', 'UI:OnlyForThisList' => '仅这个列表', 'UI:ForAllLists' => '默认所有列表', diff --git a/documentation/deprecations.md b/documentation/deprecations.md index fcc1d19b9..4ab9c91e3 100644 --- a/documentation/deprecations.md +++ b/documentation/deprecations.md @@ -1,4 +1,4 @@ -# Deprecated in 2.8.0 +# Deprecated in 3.0.0 * TabManager::GetCurrentTabLength() * TabManager::TruncateTab() @@ -6,8 +6,9 @@ * TabManager::RenderIntoContent() * iTopWebPage::SelectTab() +*cmdbabstract->GetDisplaySet +*cmdbabstract->GetDisplayExtendedSet - -# Modified in 2.8.0 +# Modified in 3.0.0 * iTabbedPage::AddTabContainer -> one optional parameter added \ No newline at end of file diff --git a/js/dataTables.pipeline.js b/js/dataTables.pipeline.js new file mode 100644 index 000000000..eb4e8398e --- /dev/null +++ b/js/dataTables.pipeline.js @@ -0,0 +1,125 @@ +// +// Pipelining function for DataTables. To be used to the `ajax` option of DataTables +// +$.fn.dataTable.pipeline = function ( opts ) { + // Configuration options + var conf = $.extend( { + pages: 5, // number of pages to cache + url: '', // script url + data: null, // function or object with parameters to send to the server + // matching how `ajax.data` works in DataTables + method: 'GET' // Ajax HTTP method + }, opts ); + + // Private variables for storing the cache + var cacheLower = -1; + var cacheUpper = null; + var cacheLastRequest = null; + var cacheLastJson = null; + var draw_number = 1; + + return function ( request, drawCallback, settings ) { + var ajax = false; + var requestStart = request.start; + var drawStart = request.start; + var requestLength = request.length; + if(request.start=undefined) + { + requestStart = settings._iDisplayStart; + drawStart = settings._iDisplayStart; + requestLength = settings._iDisplayLength; + } + var requestEnd = requestStart + requestLength; + + if ( settings.clearCache ) { + // API requested that the cache be cleared + ajax = true; + settings.clearCache = false; + } + else if ( cacheLower < 0 || requestStart < cacheLower || requestEnd > cacheUpper ) { + // outside cached data - need to make a request + ajax = true; + } + else if ( JSON.stringify( request.order ) !== JSON.stringify( cacheLastRequest.order ) || + JSON.stringify( request.columns ) !== JSON.stringify( cacheLastRequest.columns ) || + JSON.stringify( request.search ) !== JSON.stringify( cacheLastRequest.search ) + ) { + // properties changed (ordering, columns, searching) + ajax = true; + } + else if(cacheLastJson == undefined || cacheLastJson.length==0) + { + ajax = true; + } + + // Store the request for checking next time around + cacheLastRequest = $.extend( true, {}, request ); + + if ( ajax ) { + // Need data from the server + if ( requestStart < cacheLower ) { + requestStart = requestStart - (requestLength*(conf.pages-1)); + + if ( requestStart < 0 ) { + requestStart = 0; + } + } + + cacheLower = requestStart; + cacheUpper = requestStart + (requestLength * conf.pages); + + request.start = requestStart; + request.length = requestLength*conf.pages; + request.end = requestStart+ requestLength*conf.pages; + + // Provide the same `data` options as DataTables. + if ( typeof conf.data === 'function' ) { + // As a function it is executed with the data object as an arg + // for manipulation. If an object is returned, it is used as the + // data object to submit + var d = conf.data( request ); + if ( d ) { + $.extend( request, d ); + } + } + else if ( $.isPlainObject( conf.data ) ) { + // As an object, the data given extends the default + $.extend( request, conf.data ); + } + return $.ajax( { + "type": conf.method, + "url": conf.url, + "data": request, + "dataType": "json", + "cache": false, + "success": function ( json ) { + cacheLastJson = $.extend(true, {}, json); + + if ( cacheLower != drawStart ) { + json.data.splice( 0, drawStart-cacheLower ); + } + if ( requestLength >= -1 ) { + json.data.splice( requestLength, json.data.length ); + } + drawCallback( json ); + } + } ); + } + else { + json = $.extend( true, {}, cacheLastJson ); + json.draw = request.draw; // Update the echo for each response + json.data.splice( 0, requestStart-cacheLower ); + json.data.splice( requestLength, json.data.length ); + + drawCallback(json); + } + } +}; + +// Register an API method that will empty the pipelined data, forcing an Ajax +// fetch on the next draw (i.e. `table.clearPipeline().draw()`) +$.fn.dataTable.Api.register( 'clearPipeline()', function () { + return this.iterator( 'table', function ( settings ) { + settings.clearCache = true; + } ); +} ); \ No newline at end of file diff --git a/js/dataTables.settings.js b/js/dataTables.settings.js new file mode 100644 index 000000000..48a20a30f --- /dev/null +++ b/js/dataTables.settings.js @@ -0,0 +1,334 @@ +// jQuery UI style "widget" for selecting and sorting "fields" +$(function() +{ + // the widget definition, where "itop" is the namespace, + // "DataTableSettings" the widget name + $.widget( "itop.DataTableSettings", +{ + // default options + options: + { + sListId: '', + oColumns: {}, + sSelectMode: '', + sViewLink: 'true', + iPageSize: -1, + oClassAliases: {}, + sTableId : null, + oExtraParams: {}, + sRenderUrl: 'index.php', + oRenderParameters: {}, + oDefaultSettings: {}, + oLabels: { moveup: 'Move Up', movedown: 'Move Down' } + }, + + // the constructor + _create: function(mydatatable, options) + { + this.aDlgStateParams = ['iDefaultPageSize', 'oColumns']; + this.element.addClass('itop-datatable'); + + var me = this; + var bViewLink = (this.options.sViewLink == 'true'); + $('#sfl_'+me.options.sListId).fieldsorter({hasKeyColumn: bViewLink, labels: this.options.oLabels, fields: this.options.oColumns, onChange: function() { me._onSpecificSettings(); } }); + $('#datatable_dlg_'+me.options.sListId).find('input[name=page_size]').click(function() { me._onSpecificSettings(); }); + $('#datatable_dlg_'+me.options.sListId).find('input[name=save_settings]').click(function() { me._updateSaveScope(); }); + this.element.find('.itop_popup > ul li').popupmenu(); + this._updateSaveScope(); + this._saveDlgState(); + }, + + // called when created, and later when changing options + _refresh: function() + { + oParams = this.options.oData; + oParams.operation = 'search_and_refresh'; + + oParams.start = 0; + oParams.end = this.options.iPageSize; + oParams.select_mode = this.options.sSelectMode; + oParams.display_key = this.options.sViewLink; + oParams.class_aliases = this.options.oClassAliases; + oParams.columns = this.options.oColumns; + var iSortCol = 0; + var aCurrentSort = []; + for(var k1 in oParams.columns) //Aliases + { + for(var k2 in oParams.columns[k1]) //Attribute codes + { + if (oParams.columns[k1][k2].sort != 'none') + { + oParams.sort_col = iSortCol; + oParams.sort_order = oParams.columns[k1][k2].sort; + aCurrentSort.push([iSortCol, (oParams.columns[k1][k2].sort == 'asc') ? 0 : 1]); + break; //TODO make this more generic, Sort on just one column for now + } + iSortCol++; + } + break; //TODO: DBObjectSet supports only sorting on the first alias of the set + } + oParams.list_id = this.options.sListId; + var me = this; + this.element.block(); + + $('#'+me.options.sListId).DataTable().ajax.reload(); + $.post(this.options.sRenderUrl, oParams, function(data) { + // Nasty workaround to clear the pager's state for paginated lists !!! + // See jquery.tablesorter.pager.js / saveParams / restoreParams + if (window.pager_params) + { + window.pager_params['pager'+me.options.sListId] = undefined; + } + + // try { + var parentElt = $('#'+me.options.sListId).parent().parent(); + $('#'+me.options.sListId).DataTable().destroy(true); + var sThead=""; + if(me.options.sSelectMode !=""){ + sThead += ""; + } + var aOptions = JSON.parse(data); + $.each(aOptions['allColumns'], function(i, item) { + $.each(item, function(j, champs) { + if(champs.checked == 'true') { + sThead += ""+champs.label+""; + } + }); + }); + $.each(aOptions['columns'], function(i, item) { + aOptions["columns"][i]["render"]["display"] = new Function ( "data, type, row" , aOptions["columns"][i]["render"]["display"]); + }); + + parentElt.append( "" + + ""+sThead+"
" ); + aOptions["lengthMenu"]= [[oParams.end, oParams.end*2, oParams.end*3, oParams.end*4, -1], [oParams.end, oParams.end*2, oParams.end*3, oParams.end*4, aOptions["lengthMenu"]]]; + aOptions["ajax"]=eval(aOptions["ajax"]); + $('#'+me.options.sListId).DataTable(aOptions); + + me.element.unblock(); + + }, 'html' ); + + }, + _useDefaultSettings: function(bResetAll) + { + var oParams = this.options.oData; + oParams.operation = 'datatable_reset_settings'; + + oParams.table_id = this.options.sTableId; + oParams.defaults = bResetAll; + oParams.class_aliases = this.options.oClassAliases; + + var me = this; + $.post(this.options.sRenderUrl, oParams, function(data) { + // Do nothing... + }, 'html' ); + }, + _saveSettings: function(bSaveAsDefaults) + { + var oParams = this.options.oData ; + oParams.operation = 'datatable_save_settings'; + oParams.page_size = this.options.iPageSize; + oParams.table_id = this.options.sTableId; + oParams.defaults = bSaveAsDefaults; + oParams.columns = this.options.oColumns; + var iSortCol = 0; + var sSortOrder = ''; + for(var i in this.options.oColumns) + { + if (this.options.oColumns[i].checked) + { + if (this.options.oColumns[i].sort != 'none') + { + sSortOrder = this.options.oColumns[i].sort; + } + else + { + iSortCol++; + } + } + } + oParams.sort_col = iSortCol; + oParams.sort_order = sSortOrder; + var me = this; + $.post(this.options.sRenderUrl, oParams, function(data) { + // Do nothing... + }, 'html' ); + }, + onDlgOk: function() + { + var oOptions = {}; + oSettings = $('#datatable_dlg_'+this.options.sListId).find('input[name=settings]:checked'); + if (oSettings.val() == 'defaults') + { + oOptions = { iPageSize: this.options.oDefaultSettings.iDefaultPageSize, + oColumns: this.options.oDefaultSettings.oColumns + }; + } + else + { + var oDisplayColumns = {}; + var iColIdx = 0; + var iSortIdx = 0; + var sSortDirection = 'asc'; + var oColumns = $('#datatable_dlg_'+this.options.sListId).find(':itop-fieldsorter').fieldsorter('get_params'); + var iPageSize = parseInt($('#datatable_dlg_'+this.options.sListId+' input[name=page_size]').val(), 10); + + oOptions = {oColumns: oColumns, iPageSize: iPageSize, iDefaultPageSize: iPageSize }; + } + this._setOptions(oOptions); + this._refresh(); + + // Check if we need to save the settings or not... + var oSaveCheck = $('#datatable_dlg_'+this.options.sListId).find('input[name=save_settings]'); + var oSaveScope = $('#datatable_dlg_'+this.options.sListId).find('input[name=scope]:checked'); + if (oSaveCheck.prop('checked')) + { + if (oSettings.val() == 'defaults') + { + this._useDefaultSettings((oSaveScope.val() == 'defaults')); + } + else + { + this._saveSettings((oSaveScope.val() == 'defaults')); + } + } + this._saveDlgState(); + + }, + onDlgCancel: function() + { + this._restoreDlgState(); + }, + _onSpecificSettings: function() + { + $('#datatable_dlg_'+this.options.sListId).find('input.specific_settings').prop('checked', true); + }, + _updateSaveScope: function() + { + var oSaveCheck = $('#datatable_dlg_'+this.options.sListId).find('input[name=save_settings]'); + if (oSaveCheck.prop('checked')) + { + $('#datatable_dlg_'+this.options.sListId).find('input[name=scope]').each(function() { + if ($(this).attr('stay-disabled') != 'true') + { + $(this).prop('disabled', false); + } + }); + } + else + { + $('#datatable_dlg_'+this.options.sListId).find('input[name=scope]').prop('disabled', true); + } + }, + // events bound via _bind are removed automatically + // revert other modifications here + _destroy: function() + { + this.element.removeClass('itop-datatable'); + + $('#sfl_'+this.options.sListId).remove(); + $('#datatable_dlg_'+this.options.sListId).remove(); + }, + // _setOptions is called with a hash of all options that are changing + _setOptions: function() + { + // in 1.9 would use _superApply + this._superApply(arguments); + }, + // _setOption is called for each individual option that is changing + _setOption: function( key, value ) + { + // in 1.9 would use _super + this._superApply(arguments); + }, + UpdateState: function( config ) + { + var iPageSize = config.page_size; + if (iPageSize == -1) + { + iPageSize = 0; + } + this.options.iPageSize = iPageSize; + + var iPos = 0; + for (alias in this.options.oColumns) + { + for (attcode in this.options.oColumns[alias]) + { + this.options.oColumns[alias][attcode]['sort'] = 'none'; + if (this.options.oColumns[alias][attcode]['checked']) + { + if (iPos == config.sort_index) + { + this.options.oColumns[alias][attcode]['sort'] = config.sort_order; + } + iPos++; + } + } + } + + var dlgElement = $('#datatable_dlg_'+this.options.sListId); + dlgElement.find('input[name=page_size]').val(iPageSize); + dlgElement.find(':itop-fieldsorter').fieldsorter('option', { fields: this.options.oColumns }); + }, + _saveDlgState: function() + { + this.originalState = {}; + for(k in this.aDlgStateParams) + { + this.originalState[this.aDlgStateParams[k]] = this.options[this.aDlgStateParams[k]]; + } + this.originalState.oFields = $('#datatable_dlg_'+this.options.sListId).find(':itop-fieldsorter').fieldsorter('get_params'); + }, + _restoreDlgState: function() + { + var dlgElement = $('#datatable_dlg_'+this.options.sListId); + + for(k in this.aDlgStateParams) + { + this._setOption(this.aDlgStateParams[k], this.originalState[this.aDlgStateParams[k]]); + this._refresh(); + } + + dlgElement.find('input[name=page_size]').val(this.originalState.iDefaultPageSize); + + dlgElement.find(':itop-fieldsorter').fieldsorter('option', { fields: this.originalState.oFields }); + + $('#datatable_dlg_'+this.options.sListId).unblock(); + + }, + IsDialogOpen: function() + { + var oDlgOpen = $('#datatable_dlg_'+this.options.sListId+' :visible'); + + return (oDlgOpen.length > 0); + }, + DoRefresh: function() + { + this._refresh(); + }, + GetMultipleSelectionParams: function() + { + var oRes = {}; + + oRes.selectionMode = ''; + if (this.element.find(':input[name=selectionMode]').length > 0) + { + oRes.selectionMode = this.element.find(':input[name=selectionMode]').val(); + } + + oRes.selectObject = []; + this.element.find(':input[name^=selectObject]:checked').each(function() { + oRes.selectObject.push($(this).val()); + }); + + oRes.storedSelection = []; + this.element.find(':input[name^=storedSelection]').each(function() { + oRes.storedSelection.push($(this).val()); + }); + + return oRes; + } + }); +}); \ No newline at end of file diff --git a/js/datatable.js b/js/datatable.js index 37c0efd27..d57f7c129 100644 --- a/js/datatable.js +++ b/js/datatable.js @@ -1,4 +1,5 @@ // jQuery UI style "widget" for selecting and sorting "fields" +//@deprecated $(function() { // the widget definition, where "itop" is the namespace, diff --git a/js/linkswidget.js b/js/linkswidget.js index baee7f95b..40d7d8e62 100644 --- a/js/linkswidget.js +++ b/js/linkswidget.js @@ -193,7 +193,10 @@ function LinksWidget(id, sClass, sAttCode, iInputId, sSuffix, bDuplicates, oWizH theMap['selectionMode'] = sMode; $('#fs_SearchFormToAdd_'+me.id+' :input').each( function (i) { - theMap[this.name] = this.value; + if(this.name !="") + { + theMap[this.name] = this.value; + } } ); theMap['sRemoteClass'] = theMap['class']; // swap 'class' (defined in the form) and 'remoteClass' @@ -206,10 +209,11 @@ function LinksWidget(id, sClass, sAttCode, iInputId, sSuffix, bDuplicates, oWizH theMap[this.name].push(this.value); $(this).remove(); // Remove the selection for the next time the dialog re-opens }); + theMap["storedSelection[]"]=eval("oSelectedItemsadd_"+me.id); + // Retrieve the 'filter' definition - var table = $('#ResultsToAdd_'+me.id).find('table.listResults')[0]; - theMap['filter'] = table.config.filter; - theMap['extra_params'] = table.config.extra_params; + theMap['filter'] = $(':input[name=filter]', context).val(); + theMap['extra_params'] = $(':input[name=extra_params]', context).val(); } // else // { diff --git a/js/search/search_form_criteria.js b/js/search/search_form_criteria.js index 2dd528ef0..40baf7f12 100644 --- a/js/search/search_form_criteria.js +++ b/js/search/search_form_criteria.js @@ -380,7 +380,7 @@ $(function() // Prepare base DOM structure this.element .append('
') - .append('
'); + .append('
'); // Bind events // Note: No event to handle criteria closing when clicking outside of it as it is already handle by the form handler. diff --git a/js/search/search_form_handler.js b/js/search/search_form_handler.js index 3e68900e3..dc0d4b581 100644 --- a/js/search/search_form_handler.js +++ b/js/search/search_form_handler.js @@ -203,8 +203,8 @@ $(function() $('#ibo-breadcrumbs') - .breadcrumb('destroy') - .breadcrumb({ + .breadcrumbs('destroy') + .breadcrumbs({ itop_instance_id: oData['breadcrumb_instance_id'], max_count: oData['breadcrumb_max_count'], new_entry: { @@ -348,14 +348,14 @@ $(function() me._submit(); }); // - Toggle icon - this.element.find('.sf_title').on('click', function(oEvent){ + this.element.find('.ibo-panel--header').on('click', function(oEvent){ // Prevent anchors oEvent.preventDefault(); // Prevent toggle on "; + $aColumnDefinition["type"] = "html"; + $aColumnDefinition["data"] = ""; + $aColumnDefinition["render"] = "function (data, type, row) { + var oCheckboxElem = $(''); + if (row.limited_access) { + oCheckboxElem.html('-'); + } else { + oCheckboxElem.find(':input').attr('data-object-id', row.id).attr('data-target-object-id', row.target_id); + } + return oCheckboxElem.prop('outerHTML'); + }"; + array_push($aColumnsDefinitions, $aColumnDefinition); + } + + foreach ($aColumns as $sClassName => $aClassColumns) { + $aClassAliases[$sClassName] = $sClassName; + foreach ($aClassColumns as $sAttCode => $aData) { + if ($aData['checked'] == "true") { + $aColumnDefinition["width"] = "auto"; + $aColumnDefinition["searchable"] = false; + $aColumnDefinition["sortable"] = true; + $aColumnDefinition["defaultContent"] = ""; + $aColumnDefinition["type"] = "html"; + + if ($sAttCode == '_key_') { + $aColumnDefinition["title"] = $aData['alias']; + $aColumnDefinition['metadata'] = [ + 'object_class' => $sClassName, + 'attribute_code' => $sAttCode, + 'attribute_type' => '_key_', + 'attribute_label' => $aData['alias'], + ]; + $aColumnDefinition["data"] = $sClassName."/".$sAttCode; + $aColumnDefinition["render"] = [ + "display" => "return ''+row['".$sClassName."/friendlyname']+'' ;", + "_" => $sClassName."/".$sAttCode, + ]; + } else { + $oAttDef = MetaModel::GetAttributeDef($sClassName, $sAttCode); + $sAttDefClass = get_class($oAttDef); + $sAttLabel = MetaModel::GetLabel($sClassName, $sAttCode); + + $aColumnDefinition["title"] = $sAttLabel; + $aColumnDefinition['metadata'] = [ + 'object_class' => $sClassName, + 'attribute_code' => $sAttCode, + 'attribute_type' => $sAttDefClass, + 'attribute_label' => $sAttLabel, + ]; + $aColumnDefinition["data"] = $sClassName."/".$sAttCode; + $aColumnDefinition["render"] = [ + "display" => $oAttDef->GetRenderForDataTable($sClassName), + "_" => $sClassName."/".$sAttCode, + ]; + } + array_push($aColumnsDefinitions, $aColumnDefinition); + } + } + } + + $aOptions['select'] = ["style"=>$sSelectMode]; + + $aOptions['pageLength'] = $iLength; + + $sAjaxData = json_encode([ + "operation" => 'search', + "filter" => $sFilter, + "columns" => $aColumns, + "extra_params" => $aExtraParams, + "class_aliases" => $aClassAliases, + "select_mode" => $sSelectMode, + ]); + + + $aOptions =array_merge ($aOptions, [ + "language" => + [ + "processing" => Dict::Format('UI:Datatables:Language:Processing'), + "search" => Dict::Format('UI:Datatables:Language:Search'), + "lengthMenu" => Dict::Format('UI:Datatables:Language:LengthMenu'), + "zeroRecords" => Dict::Format('UI:Datatables:Language:ZeroRecords'), + "info" => Dict::Format('UI:Datatables:Language:Info'), + "infoEmpty" => Dict::Format('UI:Datatables:Language:InfoEmpty'), + "infoFiltered" => Dict::Format('UI:Datatables:Language:InfoFiltered'), + "emptyTable" => Dict::Format('UI:Datatables:Language:EmptyTable'), + "paginate" => [ + "first" => "<<", + "previous" => "<", + "next" => ">", + "last" => ">>" + ], + "aria" => [ + "sortAscending" => Dict::Format('UI:Datatables:Language:Sort:Ascending'), + "sortDescending" => Dict::Format('UI:Datatables:Language:Sort:Descending') + ], + ], + "lengthMenu" => Dict::Format('Portal:Datatables:Language:DisplayLength:All'), + "dom" => "<'ibo-datatable-toolbar'pil>t<'ibo-datatable-toolbar'pil>", + "ordering"=>true, + "order" => [], + "filter" => false, + "processing" => true, + "serverSide" => true, + "columns" => $aColumnsDefinitions, + "allColumns" => $aColumns, + 'ajax' => '$.fn.dataTable.pipeline( { + "url": "ajax.render.php", + "data": '.$sAjaxData.', + "method": "post", + "pages": 5 // number of pages to cache + } )' + ]); + + return $aOptions; + } + + public static function MakeForStaticData(string $sTitle, array $aColumns, array $aData, ?string $sId = null) + { + $oBlock = new UIContentBlock(); + $oTitle = TitleFactory::MakeNeutral($sTitle, 3); + $oBlock->AddSubBlock($oTitle); + $oTable = new StaticTable($sId); + $oTable->SetColumns($aColumns); + $oTable->SetData($aData); + $oBlock->AddSubBlock($oTable); + + return $oBlock; + } + + public Static function MakeForForm(string $sRef, array $aColumns): FormTable + { + $oTable = new FormTable($sRef); + $oTable->SetColumns($aColumns); + + return $oTable; + } +} \ No newline at end of file diff --git a/sources/application/UI/Component/DataTable/DataTableSettings.php b/sources/application/UI/Component/DataTable/DataTableSettings.php new file mode 100644 index 000000000..d2f4ffa56 --- /dev/null +++ b/sources/application/UI/Component/DataTable/DataTableSettings.php @@ -0,0 +1,356 @@ +aClassAliases = $aClassAliases; + $this->sTableId = $sTableId; + $this->iDefaultPageSize = 10; + $this->aColumns = array(); + } + + /** + * @param $iDefaultPageSize + * @param $aSortOrder + * @param $aColumns + */ + protected function Init($iDefaultPageSize, $aSortOrder, $aColumns) + { + $this->iDefaultPageSize = $iDefaultPageSize; + $this->aColumns = $aColumns; + $this->FixVisibleColumns(); + } + + /** + * @return string + */ + public function serialize() + { + // Save only the 'visible' columns + $aColumns = array(); + foreach ($this->aClassAliases as $sAlias => $sClass) { + $aColumns[$sAlias] = array(); + foreach ($this->aColumns[$sAlias] as $sAttCode => $aData) { + unset($aData['label']); // Don't save the display name + unset($aData['alias']); // Don't save the alias (redundant) + unset($aData['code']); // Don't save the code (redundant) + if ($aData['checked']) { + $aColumns[$sAlias][$sAttCode] = $aData; + } + } + } + return serialize( + array( + 'iDefaultPageSize' => $this->iDefaultPageSize, + 'aColumns' => $aColumns, + ) + ); + } + + /** + * @param string $sData + * + * @throws \Exception + */ + public function unserialize($sData) + { + $aData = unserialize($sData); + $this->iDefaultPageSize = $aData['iDefaultPageSize']; + $this->aColumns = $aData['aColumns']; + foreach ($this->aClassAliases as $sAlias => $sClass) { + foreach ($this->aColumns[$sAlias] as $sAttCode => $aData) { + $aFieldData = false; + if ($sAttCode == '_key_') { + $aFieldData = $this->GetFieldData($sAlias, $sAttCode, null, true /* bChecked */, $aData['sort']); + } else if (MetaModel::isValidAttCode($sClass, $sAttCode)) { + $oAttDef = MetaModel::GetAttributeDef($sClass, $sAttCode); + $aFieldData = $this->GetFieldData($sAlias, $sAttCode, $oAttDef, true /* bChecked */, $aData['sort']); + } + + if ($aFieldData) { + $this->aColumns[$sAlias][$sAttCode] = $aFieldData; + } else { + unset($this->aColumns[$sAlias][$sAttCode]); + } + } + } + $this->FixVisibleColumns(); + } + + /** + * @param $aClassAliases + * @param $bViewLink + * @param $aDefaultLists + * + * @return DataTableSettings + * @throws \CoreException + * @throws \DictExceptionMissingString + */ + public static function GetDataModelSettings($aClassAliases, $bViewLink, $aDefaultLists) + { + $oSettings = new DataTableSettings($aClassAliases); + // Retrieve the class specific settings for each class/alias based on the 'list' ZList + //TODO let the caller pass some other default settings (another Zlist, extre fields...) + $aColumns = array(); + foreach ($aClassAliases as $sAlias => $sClass) { + if ($aDefaultLists == null) { + $aList = cmdbAbstractObject::FlattenZList(MetaModel::GetZListItems($sClass, 'list')); + } else { + $aList = $aDefaultLists[$sAlias]; + } + + $aSortOrder = MetaModel::GetOrderByDefault($sClass); + if ($bViewLink) { + $sSort = 'none'; + if (array_key_exists('friendlyname', $aSortOrder)) { + $sSort = $aSortOrder['friendlyname'] ? 'asc' : 'desc'; + } + $sNormalizedFName = MetaModel::NormalizeFieldSpec($sClass, 'friendlyname'); + if (array_key_exists($sNormalizedFName, $aSortOrder)) { + $sSort = $aSortOrder[$sNormalizedFName] ? 'asc' : 'desc'; + } + + $aColumns[$sAlias]['_key_'] = $oSettings->GetFieldData($sAlias, '_key_', null, true /* bChecked */, $sSort); + } + foreach ($aList as $sAttCode) { + $sSort = 'none'; + if (array_key_exists($sAttCode, $aSortOrder)) { + $sSort = $aSortOrder[$sAttCode] ? 'asc' : 'desc'; + } + $oAttDef = Metamodel::GetAttributeDef($sClass, $sAttCode); + $aFieldData = $oSettings->GetFieldData($sAlias, $sAttCode, $oAttDef, true /* bChecked */, $sSort); + if ($aFieldData) $aColumns[$sAlias][$sAttCode] = $aFieldData; + } + } + $iDefaultPageSize = appUserPreferences::GetPref('default_page_size', MetaModel::GetConfig()->GetMinDisplayLimit()); + $oSettings->Init($iDefaultPageSize, $aSortOrder, $aColumns); + return $oSettings; + } + + /** + * @throws \CoreException + */ + protected function FixVisibleColumns() + { + foreach ($this->aClassAliases as $sAlias => $sClass) { + if (!isset($this->aColumns[$sAlias])) { + continue; + } + foreach ($this->aColumns[$sAlias] as $sAttCode => $aData) { + // Remove non-existent columns + // TODO: check if the existing ones are still valid (in case their type changed) + if (($sAttCode != '_key_') && (!MetaModel::IsValidAttCode($sClass, $sAttCode))) { + unset($this->aColumns[$sAlias][$sAttCode]); + } + } + $aList = MetaModel::ListAttributeDefs($sClass); + + // Add the other (non visible ones), sorted in alphabetical order + $aTempData = array(); + foreach ($aList as $sAttCode => $oAttDef) { + if ((!array_key_exists($sAttCode, $this->aColumns[$sAlias])) && (!($oAttDef instanceof AttributeLinkedSet || $oAttDef instanceof AttributeDashboard))) { + $aFieldData = $this->GetFieldData($sAlias, $sAttCode, $oAttDef, false /* bChecked */, 'none'); + if ($aFieldData) $aTempData[$aFieldData['label']] = $aFieldData; + } + } + ksort($aTempData); + foreach ($aTempData as $sLabel => $aFieldData) { + $this->aColumns[$sAlias][$aFieldData['code']] = $aFieldData; + } + } + } + + /** + * @param $aClassAliases + * @param null $sTableId + * @param bool $bOnlyOnTable + * + * @return DataTableSettings|null + * @throws \Exception + */ + static public function GetTableSettings($aClassAliases, $sTableId = null, $bOnlyOnTable = false) + { + $pref = null; + $oSettings = new DataTableSettings($aClassAliases, $sTableId); + + if ($sTableId != null) { + // An identified table, let's fetch its own settings (if any) + $pref = appUserPreferences::GetPref($oSettings->GetPrefsKey($sTableId), null); + } + + if ($pref == null) { + if (!$bOnlyOnTable) { + // Try the global preferred values for this class / set of classes + $pref = appUserPreferences::GetPref($oSettings->GetPrefsKey(null), null); + } + if ($pref == null) { + // no such settings, use the default values provided by the data model + return null; + } + } + $oSettings->unserialize($pref); + + return $oSettings; + } + + /** + * @return array + */ + public function GetSortOrder() + { + $aSortOrder = array(); + foreach ($this->aColumns as $sAlias => $aColumns) { + foreach ($aColumns as $aColumn) { + if ($aColumn['sort'] != 'none') { + $sCode = ($aColumn['code'] == '_key_') ? 'friendlyname' : $aColumn['code']; + $aSortOrder[$sCode] = ($aColumn['sort'] == 'asc'); // true for ascending, false for descending + } + } + break; // TODO: For now the Set object supports only sorting on the first class of the set + } + return $aSortOrder; + } + + /** + * @param null $sTargetTableId + * + * @return bool + */ + public function Save($sTargetTableId = null) + { + $sSaveId = is_null($sTargetTableId) ? $this->sTableId : $sTargetTableId; + if ($sSaveId == null) return false; // Cannot save, the table is not identified, use SaveAsDefault instead + + $sSettings = $this->serialize(); + appUserPreferences::SetPref($this->GetPrefsKey($sSaveId), $sSettings); + return true; + } + + /** + * @return bool + */ + public function SaveAsDefault() + { + $sSettings = $this->serialize(); + appUserPreferences::SetPref($this->GetPrefsKey(null), $sSettings); + return true; + } + + + /** + * Clear the preferences for this particular table + * @param $bResetAll boolean If true,the settings for all tables of the same class(es)/alias(es) are reset + */ + public function ResetToDefault($bResetAll) + { + if (($this->sTableId == null) && (!$bResetAll)) return false; // Cannot reset, the table is not identified, use force $bResetAll instead + if ($bResetAll) { + // Turn the key into a suitable PCRE pattern + $sKey = $this->GetPrefsKey(null); + $sPattern = str_replace(array('|'), array('\\|'), $sKey); // escape the | character + $sPattern = '#^' . str_replace(array('*'), array('.*'), $sPattern) . '$#'; // Don't use slash as the delimiter since it's used in our key to delimit aliases + appUserPreferences::UnsetPref($sPattern, true); + } else { + appUserPreferences::UnsetPref($this->GetPrefsKey($this->sTableId), false); + } + return true; + } + + /** + * @param null $sTableId + * + * @return string + */ + protected function GetPrefsKey($sTableId = null) + { + return static::GetAppUserPreferenceKey($this->aClassAliases, $sTableId); + } + + public static function GetAppUserPreferenceKey($aClassAliases, $sTableId) + { + if ($sTableId === null) { + $sTableId = '*'; + } + + $aKeys = array(); + foreach ($aClassAliases as $sAlias => $sClass) { + $aKeys[] = $sAlias . '-' . $sClass; + } + return implode('/', $aKeys) . '|' . $sTableId; + } + + /** + * @param $sAlias + * @param $sAttCode + * @param $oAttDef + * @param $bChecked + * @param $sSort + * + * @return array|bool + * @throws \CoreException + * @throws \DictExceptionMissingString + */ + protected function GetFieldData($sAlias, $sAttCode, $oAttDef, $bChecked, $sSort) + { + $ret = false; + if ($sAttCode == '_key_') { + $sLabel = Dict::Format('UI:ExtKey_AsLink', MetaModel::GetName($this->aClassAliases[$sAlias])); + $ret = array( + 'label' => $sLabel, + 'checked' => true, + 'disabled' => true, + 'alias' => $sAlias, + 'code' => $sAttCode, + 'sort' => $sSort, + ); + } else if (!$oAttDef->IsLinkSet()) { + $sLabel = $oAttDef->GetLabel(); + if ($oAttDef->IsExternalKey()) { + $sLabel = Dict::Format('UI:ExtKey_AsLink', $oAttDef->GetLabel()); + } else if ($oAttDef->IsExternalField()) { + if ($oAttDef->IsFriendlyName()) { + $sLabel = Dict::Format('UI:ExtKey_AsFriendlyName', $oAttDef->GetLabel()); + } else { + $oExtAttDef = $oAttDef->GetExtAttDef(); + $sLabel = Dict::Format('UI:ExtField_AsRemoteField', $oAttDef->GetLabel(), $oExtAttDef->GetLabel()); + } + } elseif ($oAttDef instanceof AttributeFriendlyName) { + $sLabel = Dict::Format('UI:ExtKey_AsFriendlyName', $oAttDef->GetLabel()); + } + $ret = array( + 'label' => $sLabel, + 'checked' => $bChecked, + 'disabled' => false, + 'alias' => $sAlias, + 'code' => $sAttCode, + 'sort' => $sSort, + ); + } + return $ret; + } +} \ No newline at end of file diff --git a/sources/application/UI/Component/DataTable/StaticTable/FormTable/FormTable.php b/sources/application/UI/Component/DataTable/StaticTable/FormTable/FormTable.php new file mode 100644 index 000000000..a535d6acd --- /dev/null +++ b/sources/application/UI/Component/DataTable/StaticTable/FormTable/FormTable.php @@ -0,0 +1,65 @@ +SetRef($sRef); + $this->aRows = []; + } + + /** + * @return string + */ + public function GetRef(): string + { + return $this->sRef; + } + + /** + * @param string $sRef + */ + public function SetRef(string $sRef): void + { + $this->sRef = $sRef; + } + + public function GetRows(): array + { + return $this->aRows; + } + + public function AddRow(FormTableRow $oRow): self + { + $this->aRows[] = $oRow; + return $this; + } +} \ No newline at end of file diff --git a/sources/application/UI/Component/DataTable/StaticTable/FormTableRow/FormTableRow.php b/sources/application/UI/Component/DataTable/StaticTable/FormTableRow/FormTableRow.php new file mode 100644 index 000000000..01dfd3f84 --- /dev/null +++ b/sources/application/UI/Component/DataTable/StaticTable/FormTableRow/FormTableRow.php @@ -0,0 +1,124 @@ + [ + * 'description' => tooltip, + * 'label' => label to display, + * 'class' => cell CSS class, + * 'metadata' => [key => value] transformed into data-key="value" + * ], ... + * ] + */ + private $aColumns; + + /** + * @var array + * [ + * 'entry name' => [ + * 'value_html' => value to display in the cell, + * 'value_raw' => real value put into data-value-raw + * ], ... + * ] + */ + private $aData; + + public function __construct(string $sRef, array $aColumns, array $aData, int $iRowId) + { + parent::__construct(); + $this->SetRef($sRef); + $this->SetColumns($aColumns); + $this->SetData($aData); + $this->SetRowId($iRowId); + } + + /** + * @return string + */ + public function GetRef(): string + { + return $this->sRef; + } + + /** + * @param string $sRef + */ + public function SetRef(string $sRef): void + { + $this->sRef = $sRef; + } + + /** + * @return array + */ + public function GetColumns(): array + { + return $this->aColumns; + } + + /** + * @param array $aColumns + */ + public function SetColumns(array $aColumns): void + { + $this->aColumns = $aColumns; + } + + /** + * @return array + */ + public function GetData(): array + { + return $this->aData; + } + + /** + * @param array $aData + */ + public function SetData(array $aData): void + { + $this->aData = $aData; + } + + /** + * @return int + */ + public function GetRowId(): int + { + return $this->iRowId; + } + + /** + * @param int $iRowId + */ + public function SetRowId(int $iRowId): void + { + $this->iRowId = $iRowId; + } +} \ No newline at end of file diff --git a/sources/application/UI/Component/DataTable/StaticTable/StaticTable.php b/sources/application/UI/Component/DataTable/StaticTable/StaticTable.php new file mode 100644 index 000000000..64aaa5a55 --- /dev/null +++ b/sources/application/UI/Component/DataTable/StaticTable/StaticTable.php @@ -0,0 +1,83 @@ + [ + * 'description' => tooltip, + * 'label' => label to display, + * 'class' => cell CSS class, + * 'metadata' => [key => value] transformed into data-key="value" + * ] + */ + private $aColumns; + + /** + * @var array of [ + * '@class' => css class of the row, + * 'entry name' => [ + * 'value_html' => value to display in the cell, + * 'value_raw' => real value put into data-value-raw + * ], ... + * ] + */ + private $aData; + + public function __construct(string $sId = null, string $sContainerCSSClass = '') + { + parent::__construct($sId, $sContainerCSSClass); + $this->aColumns = []; + $this->aData = []; + } + + /** + * @return array + */ + public function GetColumns(): array + { + return $this->aColumns; + } + + /** + * @param array $aColumns + */ + public function SetColumns(array $aColumns): void + { + $this->aColumns = $aColumns; + } + + /** + * @return array + */ + public function GetData(): array + { + return $this->aData; + } + + /** + * @param array $aData + */ + public function SetData(array $aData): void + { + $this->aData = $aData; + } + +} \ No newline at end of file diff --git a/sources/application/UI/Component/Title/TitleFactory.php b/sources/application/UI/Component/Title/TitleFactory.php index d810bf2e5..64b964673 100644 --- a/sources/application/UI/Component/Title/TitleFactory.php +++ b/sources/application/UI/Component/Title/TitleFactory.php @@ -66,4 +66,9 @@ class TitleFactory return $oTitle; } + + public static function MakeNeutral(string $sTitle, int $iLevel = 1, ?string $sId = null) + { + return new Title($sTitle, $iLevel, $sId); + } } \ No newline at end of file diff --git a/sources/application/WebPage/NiceWebPage.php b/sources/application/WebPage/NiceWebPage.php index da9896b11..a3c9931c7 100644 --- a/sources/application/WebPage/NiceWebPage.php +++ b/sources/application/WebPage/NiceWebPage.php @@ -40,13 +40,31 @@ class NiceWebPage extends WebPage $this->add_linked_script(utils::GetAbsoluteUrlAppRoot().'js/jquery-ui.custom.min.js'); $this->add_linked_script(utils::GetAbsoluteUrlAppRoot().'js/utils.js'); $this->add_linked_script(utils::GetAbsoluteUrlAppRoot().'js/hovertip.js'); + $this->add_linked_script(utils::GetAbsoluteUrlAppRoot().'js/table-selectable-lines.js'); + $this->add_linked_script(utils::GetAbsoluteUrlAppRoot().'js/field_sorter.js'); + //TODO deprecated in 3.0.0 + $this->add_linked_script(utils::GetAbsoluteUrlAppRoot().'js/datatable.js'); // table sorting $this->add_linked_script(utils::GetAbsoluteUrlAppRoot().'js/jquery.tablesorter.js'); $this->add_linked_script(utils::GetAbsoluteUrlAppRoot().'js/jquery.tablesorter.pager.js'); $this->add_linked_script(utils::GetAbsoluteUrlAppRoot().'js/jquery.tablehover.js'); - $this->add_linked_script(utils::GetAbsoluteUrlAppRoot().'js/table-selectable-lines.js'); - $this->add_linked_script(utils::GetAbsoluteUrlAppRoot().'js/field_sorter.js'); - $this->add_linked_script(utils::GetAbsoluteUrlAppRoot().'js/datatable.js'); + //TODO end deprecated in 3.0.0 + // Datatables added in 3.0.0 + $this->add_linked_script(utils::GetAbsoluteUrlAppRoot().'lib/datatables/js/jquery.dataTables.min.js'); + $this->add_linked_script(utils::GetAbsoluteUrlAppRoot().'lib/datatables/js/dataTables.bootstrap.min.js'); + $this->add_linked_script(utils::GetAbsoluteUrlAppRoot().'lib/datatables/js/dataTables.fixedHeader.min.js'); + $this->add_linked_script(utils::GetAbsoluteUrlAppRoot().'lib/datatables/js/dataTables.responsive.min.js'); + $this->add_linked_script(utils::GetAbsoluteUrlAppRoot().'lib/datatables/js/dataTables.scroller.min.js'); + $this->add_linked_script(utils::GetAbsoluteUrlAppRoot().'lib/datatables/js/dataTables.select.min.js'); + $this->add_linked_script(utils::GetAbsoluteUrlAppRoot().'js/dataTables.settings.js'); + $this->add_linked_script(utils::GetAbsoluteUrlAppRoot().'js/dataTables.pipeline.js'); + /*$this->add_linked_stylesheet(utils::GetAbsoluteUrlAppRoot().'lib/datatables/css/dataTables.bootstrap.min.css'); + $this->add_linked_stylesheet(utils::GetAbsoluteUrlAppRoot().'lib/datatables/css/fixedHeader.bootstrap.min.css'); + $this->add_linked_stylesheet(utils::GetAbsoluteUrlAppRoot().'lib/datatables/css/responsive.bootstrap.min.css'); + $this->add_linked_stylesheet(utils::GetAbsoluteUrlAppRoot().'lib/datatables/css/scroller.bootstrap.min.css'); + $this->add_linked_stylesheet(utils::GetAbsoluteUrlAppRoot().'lib/datatables/css/select.bootstrap.min.css'); + $this->add_linked_stylesheet(utils::GetAbsoluteUrlAppRoot().'lib/datatables/css/select.dataTables.min.css');*/ + $this->add_linked_script(utils::GetAbsoluteUrlAppRoot().'js/jquery.positionBy.js'); $this->add_linked_script(utils::GetAbsoluteUrlAppRoot().'js/jquery.popupmenu.js'); $this->add_linked_script(utils::GetAbsoluteUrlAppRoot().'js/searchformforeignkeys.js'); @@ -69,52 +87,7 @@ class NiceWebPage extends WebPage $this->add_linked_script(utils::GetAbsoluteUrlAppRoot().'js/clipboardwidget.js'); $this->add_dict_entries('UI:Combo'); - - $this->add_ready_script( -<<< EOF - //add new widget called TruncatedList to properly display truncated lists when they are sorted - $.tablesorter.addWidget({ - // give the widget a id - id: "truncatedList", - // format is called when the on init and when a sorting has finished - format: function(table) - { - // Check if there is a "truncated" line - this.truncatedList = false; - if ($("tr td.truncated",table).length > 0) - { - this.truncatedList = true; - } - if (this.truncatedList) - { - $("tr td",table).removeClass('truncated'); - $("tr:last td",table).addClass('truncated'); - } - } - }); - - $.tablesorter.addWidget({ - // give the widget a id - id: "myZebra", - // format is called when the on init and when a sorting has finished - format: function(table) - { - // Replace the 'red even' lines by 'red_even' since most browser do not support 2 classes selector in CSS, etc.. - $("tbody tr:even",table).addClass('even'); - $("tbody tr.red:even",table).removeClass('red').removeClass('even').addClass('red_even'); - $("tbody tr.orange:even",table).removeClass('orange').removeClass('even').addClass('orange_even'); - $("tbody tr.green:even",table).removeClass('green').removeClass('even').addClass('green_even'); - // In case we sort again the table, we need to remove the added 'even' classes on odd rows - $("tbody tr:odd",table).removeClass('even'); - $("tbody tr.red_even:odd",table).removeClass('even').removeClass('red_even').addClass('red'); - $("tbody tr.orange_even:odd",table).removeClass('even').removeClass('orange_even').addClass('orange'); - $("tbody tr.green_even:odd",table).removeClass('even').removeClass('green_even').addClass('green'); - } - }); - $("table.listResults").tableHover(); // hover tables -EOF - ); - $this->LoadTheme(); + $this->LoadTheme(); $this->m_sRootUrl = $this->GetAbsoluteUrlAppRoot(); $sAbsURLAppRoot = addslashes($this->m_sRootUrl); diff --git a/sources/application/search/searchform.class.inc.php b/sources/application/search/searchform.class.inc.php index 86da53990..069059055 100644 --- a/sources/application/search/searchform.class.inc.php +++ b/sources/application/search/searchform.class.inc.php @@ -31,6 +31,11 @@ use AttributeFriendlyName; use AttributeTagSet; use CMDBObjectSet; use Combodo\iTop\Application\Search\CriterionConversion\CriterionToSearchForm; +use Combodo\iTop\Application\UI\Component\Form\Form; +use Combodo\iTop\Application\UI\Component\Html\Html; +use Combodo\iTop\Application\UI\Component\Input\InputFactory; +use Combodo\iTop\Application\UI\Component\Panel\Panel; +use Combodo\iTop\Application\UI\Layout\UIContentBlock; use CoreException; use DBObjectSearch; use DBObjectSet; @@ -42,9 +47,9 @@ use IssueLog; use MetaModel; use MissingQueryArgument; use TrueExpression; +use UserRights; use utils; use WebPage; -use UserRights; class SearchForm { @@ -54,13 +59,16 @@ class SearchForm * @param array $aExtraParams * * @return string - * @throws \CoreException - * @throws \DictExceptionMissingString - * @throws \Exception */ public function GetSearchForm(WebPage $oPage, CMDBObjectSet $oSet, $aExtraParams = array()) { - $sHtml = ''; + $oPage->AddUiBlock($this->GetSearchFormUIBlock($oPage, $oSet, $aExtraParams)); + return ''; + } + + public function GetSearchFormUIBlock(WebPage $oPage, DBObjectSet $oSet, $aExtraParams = array()) + { + $oUiBlock = new UIContentBlock(); $oAppContext = new ApplicationContext(); $sClassName = $oSet->GetFilter()->GetClass(); $aListParams = array(); @@ -79,7 +87,7 @@ class SearchForm { $iSearchFormId = $oPage->GetUniqueId(); $sSearchFormId = 'SimpleSearchForm'.$iSearchFormId; - $sHtml .= "
\n"; + $oUiBlock->AddHtml("
"); $aListParams['currentId'] = "$iSearchFormId"; } // Check if the current class has some sub-classes @@ -109,7 +117,7 @@ class SearchForm if (!isset($aExtraParams['result_list_outer_selector'])) { - if (isset($aExtraParams['table_id'])) + if (isset($aExtraParams['table_id']) ) { $aExtraParams['result_list_outer_selector'] = $aExtraParams['table_id']; } @@ -168,23 +176,42 @@ class SearchForm } } } + $bShowObsoleteData = \appUserPreferences::GetPref('show_obsolete_data', MetaModel::GetConfig()->Get('obsolescence.show_obsolete_data'));// ? What to do when true == utils::IsArchiveMode() + $sAction = (isset($aExtraParams['action'])) ? $aExtraParams['action'] : utils::GetAbsoluteUrlAppRoot().'pages/UI.php'; - $sStyle = ($bOpen == 'true') ? '' : 'closed'; + $sStyle = "ibo-search-form"; + $sStyle .= ($bOpen == 'true') ? '' : ' closed'; $sStyle .= ($bAutoSubmit === true) ? '' : ' no_auto_submit'; - $sHtml .= "
\n"; // Don't use $_SERVER['SCRIPT_NAME'] since the form may be called asynchronously (from ajax.php) - $sHtml .= "

" . Dict::Format('UI:SearchFor_Class_Objects', $sClassesCombo) . "" . Dict::S('UI:SearchToggle') . ""; - $sHtml .= ""; - $sHtml .= ""; - $sHtml .= "" . Dict::S('UI:Search:Obsolescence:DisabledHint') . ""; +//(string $sTitle = '', array $aSubBlocks = [], string $sColor = self::DEFAULT_COLOR, ?string $sId = null) + $oUiSearchBlock = new Panel(Dict::Format('UI:SearchFor_Class_Objects', $sClassesCombo), [],Panel::DEFAULT_COLOR, $sSearchFormId); + $oUiSearchBlock->SetCSSClasses("display_block"); + $oUiBlock->AddSubBlock($oUiSearchBlock); + $sHtml = ""; + if ($bShowObsoleteData) + { + $sHtml .= ""; + $sHtml .= "" . Dict::S('UI:Search:Obsolescence:DisabledHint') . ""; + } + if($bAutoSubmit === false) { + $sHtml .= "
"; + $sHtml .= "".Dict::S('UI:Search:AutoSubmit:DisabledHint').""; + $sHtml .= "
"; + } $sHtml .= "
"; - $sHtml .= "" . Dict::S('UI:Search:AutoSubmit:DisabledHint') . ""; - $sHtml .= "
"; - $sHtml .= "
"; - $sHtml .= "

\n"; - $sHtml .= "
\n"; - $sHtml .= "
\n
\n"; - $sHtml .= "
\n"; + $oUiSearchBlock->AddToolbarBlock(new Html($sHtml)); + + + + $oFormSearch=new Form("fs_".$sSearchFormId); + $oFormSearch->SetAction($sAction) + ->AddCSSClasses($sStyle); + $oUiSearchBlock->AddSubBlock($oFormSearch); + $oFormSearch->AddSubBlock(InputFactory::MakeForHidden("class", $sClassName)); + $oFormSearch->AddHtml( "
");//class sf_message header_message + + $oCriterionBlock = new UIContentBlock("fs_{$sSearchFormId}_criterion_outer","sf_criterion_area ibo-criterion-area"); + $oFormSearch->AddSubBlock($oCriterionBlock); if (isset($aExtraParams['query_params'])) { @@ -256,8 +283,6 @@ class SearchForm $sDateFormat = substr($sDateTimeFormat, 0, $iDateTimeSeparatorPos); $sTimeFormat = substr($sDateTimeFormat, $iDateTimeSeparatorPos + 1); - $bShowObsoleteData = \appUserPreferences::GetPref('show_obsolete_data', MetaModel::GetConfig()->Get('obsolescence.show_obsolete_data'));// ? What to do when true == utils::IsArchiveMode() - $aSearchParams = array( 'criterion_outer_selector' => "#fs_{$sSearchFormId}_criterion_outer", 'result_list_outer_selector' => "#{$aExtraParams['result_list_outer_selector']}", @@ -289,9 +314,8 @@ class SearchForm $oPage->add_ready_script('$("#fs_'.$sSearchFormId.'").search_form_handler('.json_encode($aSearchParams).');'); - return $sHtml; + return $oUiBlock; } - /** * @param \DBObjectSet $oSet * diff --git a/synchro/synchro_import.php b/synchro/synchro_import.php index d15b7be3e..613622414 100644 --- a/synchro/synchro_import.php +++ b/synchro/synchro_import.php @@ -491,7 +491,7 @@ try $iLoopTimeLimit = MetaModel::GetConfig()->Get('max_execution_time_per_loop'); $oMutex = new iTopMutex('synchro_import_'.$oDataSource->GetKey()); $oMutex->Lock(); - set_time_limit($iLoopTimeLimit); + set_time_limit(intval($iLoopTimeLimit)); foreach ($aData as $iRow => $aRow) { $sReconciliationCondition = '`primary_key` = '.CMDBSource::Quote($aRow[$iPrimaryKeyCol]); @@ -633,7 +633,7 @@ try } } $oMutex->Unlock(); - set_time_limit($iPreviousTimeLimit); + set_time_limit(intval($iPreviousTimeLimit)); if (($sOutput === 'summary') || ($sOutput === 'details')) { diff --git a/synchro/synchrodatasource.class.inc.php b/synchro/synchrodatasource.class.inc.php index eac1c4048..3c4678492 100644 --- a/synchro/synchrodatasource.class.inc.php +++ b/synchro/synchrodatasource.class.inc.php @@ -3466,7 +3466,7 @@ class SynchroExecution /** @var SynchroReplica $oReplica */ while($oReplica = $oSetToProcess->Fetch()) { - set_time_limit($iLoopTimeLimit); + set_time_limit(intval($iLoopTimeLimit)); $iLastReplicaProcessed = $oReplica->GetKey(); switch ($sDeletePolicy) { @@ -3595,7 +3595,7 @@ class SynchroExecution /** @var \SynchroReplica $oReplica */ while ($oReplica = $oSetToProcess->Fetch()) { - set_time_limit($iLoopTimeLimit); + set_time_limit(intval($iLoopTimeLimit)); $iLastReplicaProcessed = $oReplica->GetKey(); $this->m_oStatLog->AddTrace("Synchronizing replica id=$iLastReplicaProcessed."); $oReplica->Synchro($this->m_oDataSource, $this->m_aReconciliationKeys, $this->m_aAttributes, $this->m_oChange, @@ -3697,7 +3697,7 @@ class SynchroExecution /** @var SynchroReplica $oReplica */ while($oReplica = $oSetToProcess->Fetch()) { - set_time_limit($iLoopTimeLimit); + set_time_limit(intval($iLoopTimeLimit)); $iLastReplicaProcessed = $oReplica->GetKey(); $this->m_oStatLog->AddTrace('Destination object to be DELETED', $oReplica); $oReplica->DeleteDestObject($this->m_oChange, $this->m_oStatLog); diff --git a/templates/components/datatable/layout.html.twig b/templates/components/datatable/layout.html.twig new file mode 100644 index 000000000..b45e9a70b --- /dev/null +++ b/templates/components/datatable/layout.html.twig @@ -0,0 +1,61 @@ +{% for oSubBlock in oUIBlock.GetSubBlocks() %}{{ render_block(oSubBlock, {aPage: aPage}) }}{% endfor %} +{% if oUIBlock.GetOptions()["select_mode"] is defined %} + + + +{% endif %} + + + + {% if oUIBlock.GetOptions()["select_mode"] is defined %} + + {% endif %} + {% for aColumn in oUIBlock.GetDisplayColumns() %} + + {% endfor %} + +
{{ aColumn["attribute_label"] }}
+ + \ No newline at end of file diff --git a/templates/components/datatable/layout.js.twig b/templates/components/datatable/layout.js.twig new file mode 100644 index 000000000..3fa99d332 --- /dev/null +++ b/templates/components/datatable/layout.js.twig @@ -0,0 +1,257 @@ +function checkAllDataTable(table, value) { + // Set the 'selectionMode' for the future objects to load + var selectionMode = 'positive'; + if (value) { + selectionMode = 'negative'; + } + + // Mark all the displayed items as check or unchecked depending on the value + $(table).find(':checkbox[name^=selectObj]:not([disabled])').each(function () { + var $currentCheckbox = $(this); + $currentCheckbox.prop('checked', value); + $currentLine = $currentCheckbox.closest("tr"); + (value) ? $currentLine.addClass("selected") : $currentLine.removeClass("selected"); + }); + + $(table).parent().parent().find(':input[name=selectionMode]').val(selectionMode); + // Reset the list of saved selection... + $(':input[name^=storedSelection]').remove(); + $(table).parent().find(':checkbox[name^=selectObj]').trigger("change"); + + if (value) { + $(table).DataTable().rows().select(); + } else { + $(table).DataTable().rows({page: 'current'}).deselect(); + } + + return true; +} + +$('#{{ oUIBlock.GetId() }}').closest("[role=dialog]").on("dialogbeforeclose", function () { + $('#{{ oUIBlock.GetId() }}').DataTable().clear(); +}); + +{% if oUIBlock.GetOption("select_mode") is defined %} +var oSelectedItems{{ oUIBlock.GetOption('sTableId') }} = []; +{% endif %} + +if ($.fn.dataTable.isDataTable('#{{ oUIBlock.GetId() }}')) { + $('#{{ oUIBlock.GetId() }}').DataTable().destroy(false); +} + +var oTable{{ oUIBlock.GetId() }} = $('#{{ oUIBlock.GetId() }}').DataTable({ + "language": { + "processing": "{{ 'UI:Datatables:Language:Processing'|dict_s }}", + "search": "{{ 'UI:Datatables:Language:Search'|dict_s }}", + "lengthMenu": "{{ 'UI:Datatables:Language:LengthMenu'|dict_s }}", + "zeroRecords": "{{ 'UI:Datatables:Language:ZeroRecords'|dict_s }}", + "info": "{{ 'UI:Datatables:Language:Info'|dict_s }}", + "infoEmpty": "{{ 'UI:Datatables:Language:InfoEmpty'|dict_s }}", + "infoFiltered": "({{ 'UI:Datatables:Language:InfoFiltered'|dict_s }})", + "emptyTable": "{{ 'UI:Datatables:Language:EmptyTable'|dict_s }}", + "paginate": { + "first": "<<", + previous: "", + next: "", + "last": ">>" + }, + "aria": { + "sortAscending": ": {{ 'UI:Datatables:Language:Sort:Ascending'|dict_s }}", + "sortDescending": ": {{ 'UI:Datatables:Language:Sort:Descending'|dict_s }}" + } + }, + "lengthMenu": [[ {{ oUIBlock.GetOptions()["iPageSize"] }}, {{ oUIBlock.GetOptions()["iPageSize"]*2 }}, {{ oUIBlock.GetOptions()["iPageSize"]*3 }}, {{ oUIBlock.GetOptions()["iPageSize"]*4 }}, -1], [ {{ oUIBlock.GetOptions()["iPageSize"] }}, {{ oUIBlock.GetOptions()["iPageSize"]*2 }}, {{ oUIBlock.GetOptions()["iPageSize"]*3 }}, {{ oUIBlock.GetOptions()["iPageSize"]*4 }}, "{{ 'Portal:Datatables:Language:DisplayLength:All'|dict_s }}"]], + "dom": "<'ibo-datatable-toolbar'pil>t<'ibo-datatable-toolbar'pil>", + {% if( oUIBlock.GetOptions()["sort"][0] is defined ) %} + "order": [[{{ oUIBlock.GetOptions()["sort"][0]}},'{{ oUIBlock.GetOptions()["sort"][1]}}']], + {% else %} + "order":[], + {% endif %} + "ordering": true, + {% if oUIBlock.GetOption("select_mode") is defined %} + "select": { + "style": "{{ oUIBlock.GetOption("select_mode") }}" + }, + "rowCallback": function (oRow, oData) { + // Hiding pagination if only one page + if ($(this).closest('.ibo-panel--body').find('[name=selectionMode]') === "negative") { + if (!oData.id in oSelectedItems{{ oUIBlock.GetOption('sTableId') }}) { + $(oRow).select(); + $(oRow).find('td:first-child input').prop('checked', true); + } + } else { + if (oData.id in oSelectedItems{{ oUIBlock.GetOption('sTableId') }}) { + $(oRow).select(); + $(oRow).find('td:first-child input').prop('checked', true); + } + } + }, + "drawCallback": function () { + // Hiding pagination if only one page + if ($(this).closest('.ibo-panel--body').find('[name=selectionMode]') === "negative") { + $(this).find('[name=selectAll]').checked(); + $(table).DataTable().rows({page: 'current'}).select(); + } else { + $(this).closest('.dataTables_wrapper').find('.dataTables_paginate, .dataTables_info').show(); + } + }, + {% endif %} + "rowId": "id", + "filter": false, + "retrieve": true, + "destroy": true, + "processing": true, + "serverSide": true, + "columns": [ + {% if oUIBlock.GetOption("select_mode") is not empty %} + { + "width": "auto", + "searchable": false, + "sortable": false, + "title": + {% if oUIBlock.GetOption("select_mode") != "single" %} + '' + {% else %} + '' + {% endif %}, + "type": "html", + "data": "", + "render": function (data, type, row) { + var oCheckboxElem = + {% if oUIBlock.GetOption("select_mode") != "single" %} + $(''); + {% else %} + $(''); + {% endif %} + if (row.limited_access) { + oCheckboxElem.html('-'); + } else { + oCheckboxElem.find(':input').attr('data-object-id', row.id).attr('data-target-object-id', row.target_id); + } + return oCheckboxElem.prop('outerHTML'); + } + }, + {% endif %} + {% for aColumn in oUIBlock.GetDisplayColumns() %} + { + "width": "auto", + "searchable": false, + "sortable": true, + "title": "{{ aColumn["attribute_label"] }}", + "defaultContent": "", + "type": "html", + 'metadata': { + 'object_class': "{{ aColumn["object_class"] }}", + 'attribute_code': "{{ aColumn["attribute_code"] }}", + 'attribute_type': "{{ aColumn["attribute_type"] }}", + 'attribute_label': "{{ aColumn["attribute_label"] }}" + }, + "data": "{{ aColumn["class_alias"] }}/{{ aColumn["attribute_code"] }}", + "render": { + "display": function (data, type, row) { {{ aColumn["render"]|raw }}}, + "_": "{{ aColumn["class_alias"] }}/{{ aColumn["attribute_code"] }}" + } + }, + {% endfor %} + ], + "ajax": $.fn.dataTable.pipeline({ + "url": "{{ oUIBlock.GetAjaxUrl() }}", + "data": {{ oUIBlock.GetAjaxData() |raw }}, + "method": "post", + "pages": 5 // number of pages to cache + }), + "initComplete": function() { + if (this.api().page.info().pages === 1) { + $('.dataTables_paginate').hide(); + $('.dataTables_length').hide(); + } + } +}); + +oTable{{ oUIBlock.GetId() }}.off('select').on('select', function (oEvent, dt, type, indexes) { + var aData = oTable{{ oUIBlock.GetId() }}.rows(indexes).data().toArray(); + // Checking input + $('#{{ oUIBlock.GetId() }} tr[role="row"].selected td:first-child input').prop('checked', true); + // Saving values in temp array + for (var i in aData) { + var iItemId = aData[i].id; + if (oSelectedItems{{ oUIBlock.GetOption('sTableId') }}.indexOf(iItemId) === -1) { + oSelectedItems{{ oUIBlock.GetOption('sTableId') }}.push(iItemId); + } + } + if ((oSelectedItems{{ oUIBlock.GetOption('sTableId') }}.length === oTable{{ oUIBlock.GetId() }}.page.info()["recordsTotal"] && $(this).closest('.ibo-panel--body').find('[name=selectionMode]').val() === "negative")) { + $('#btn_ok_{{ oUIBlock.GetOption('sTableId') }}').prop('disabled', true); + } else { + $('#btn_ok_{{ oUIBlock.GetOption('sTableId') }}').prop('disabled', false); + } +}); + +oTable{{ oUIBlock.GetId() }}.off('deselect').on('deselect', function (oEvent, dt, type, indexes) { + var aData = oTable{{ oUIBlock.GetId() }}.rows(indexes).data().toArray(); + + // Checking input + $('#{{ oUIBlock.GetId() }} tr[role="row"]:not(.selected) td:first-child input').prop('checked', false); + // Saving values in temp array + for (var i in aData) { + var iItemId = aData[i].id; + if (oSelectedItems{{ oUIBlock.GetOption('sTableId') }}.indexOf(iItemId) > -1) { + oSelectedItems{{ oUIBlock.GetOption('sTableId') }}.splice(oSelectedItems{{ oUIBlock.GetOption('sTableId') }}.indexOf(iItemId), 1); + } + } + if ((oSelectedItems{{ oUIBlock.GetOption('sTableId') }}.length === 0 && $(this).closest('.ibo-panel--body').find('[name=selectionMode]').val() === "positive")) { + $('#btn_ok_{{ oUIBlock.GetOption('sTableId') }}').prop('disabled', true); + } else { + $('#btn_ok_{{ oUIBlock.GetOption('sTableId') }}').prop('disabled', false); + } +}); + +// Handles submit button +//$('#form_{{ oUIBlock.GetId() }} .form_buttons .form_btn_submit').off('click').on('click', function(oEvent){ +// Extracting value(s) to be send back to the source form +{% if bMultipleSelect %} +//var oData = {values: oSelectedItems}; +{% else %} +/*var oData = {value: {}}; +var sItemId = Object.keys(oSelectedItems)[0]; +var sItemName = oSelectedItems[sItemId]; + +oData.value[sItemId] = sItemName;*/ +{% endif %} + +// Triggering value setting on the source field +//$('[data-form-path="{{ aSource.sFormPath }}"][data-field-id="{{ aSource.sFieldId }}"]').triggerHandler('set_current_value', oData); + +// Closing the modal +{% if tIsModal is defined and tIsModal == true %} +//$('#{{ sFormId }}').closest('.modal').modal('hide'); +{% endif %} +//}); + +$('#datatable_dlg_{{ oUIBlock.GetId() }}').dialog( + { + autoOpen: false, + title: "{{ 'UI:ListConfigurationTitle'|dict_s }}", + width: 500, + close: function () { //save data and refresh + $('#datatable_dlg_{{ oUIBlock.GetId() }}').DataTableSettings('onDlgCancel'); + } + }); + +$aOptions = { + 'sListId': '{{ oUIBlock.GetId() }}', + 'oColumns': {{ oUIBlock.GetResultColumnsAsJson()|raw }}, + 'sViewLink': '{{ oUIBlock.GetOptions()["bViewLink"] }}', + 'iPageSize': '{{ oUIBlock.GetOptions()["iPageSize"] }}', + 'sTableId': '{{ oUIBlock.GetOptions()["sTableId"] }}', + "sRenderUrl": "{{ oUIBlock.GetAjaxUrl() }}", + "sSelectMode": "{{ oUIBlock.GetOptions()["select_mode"] }}", + "oData": {{ oUIBlock.GetAjaxData() |raw }}, + 'oLabels': {"moveup": "{{ 'UI:Button:MoveUp'|dict_s }}", "movedown": "{{ 'UI:Button:MoveDown'|dict_s }}"}, + 'oDefaultSettings': {{ oUIBlock.GetOptions()["oDefaultSettings"]|raw }} +}; + +//if (!typeof $('#datatable_dlg_{{ oUIBlock.GetId() }}').DataTableSettings() !== "undefined") +if ($('#datatable_dlg_{{ oUIBlock.GetId() }}').DataTableSettings()) { + $('#datatable_dlg_{{ oUIBlock.GetId() }}').DataTableSettings("destroy"); +} +$('#datatable_dlg_{{ oUIBlock.GetId() }}').DataTableSettings($aOptions); \ No newline at end of file diff --git a/templates/components/datatable/static/formtable/layout.html.twig b/templates/components/datatable/static/formtable/layout.html.twig new file mode 100644 index 000000000..16e443328 --- /dev/null +++ b/templates/components/datatable/static/formtable/layout.html.twig @@ -0,0 +1,20 @@ +{# @copyright Copyright (C) 2010-2020 Combodo SARL #} +{# @license http://opensource.org/licenses/AGPL-3.0 #} + + + +{% set columns = oUIBlock.GetColumns() %} + + + + {% for column in columns %} + + {% endfor %} + + + + {% for oSubBlock in oUIBlock.GetRows() %} + {{ render_block(oSubBlock, {aPage: aPage}) }} + {% endfor %} + +
{{ column.label|raw }}
\ No newline at end of file diff --git a/templates/components/datatable/static/formtable/layout.js.twig b/templates/components/datatable/static/formtable/layout.js.twig new file mode 100644 index 000000000..8868623b2 --- /dev/null +++ b/templates/components/datatable/static/formtable/layout.js.twig @@ -0,0 +1,6 @@ +$('#{{ oUIBlock.GetId() }}').DataTable({ + language: { + emptyTable: {{ 'UI:Message:EmptyList:UseAdd'|dict_s }} + }, + search:false +}); \ No newline at end of file diff --git a/templates/components/datatable/static/formtablerow/layout.html.twig b/templates/components/datatable/static/formtablerow/layout.html.twig new file mode 100644 index 000000000..609a3e624 --- /dev/null +++ b/templates/components/datatable/static/formtablerow/layout.html.twig @@ -0,0 +1,19 @@ +{# @copyright Copyright (C) 2010-2020 Combodo SARL #} +{# @license http://opensource.org/licenses/AGPL-3.0 #} + + + {% for colName,column in oUIBlock.GetColumns() %} + + {% set cellValueHtml = '' %} + {% for cellName,cellValue in oUIBlock.GetData() %} + {% if cellName == colName %} + {% set cellValueHtml = cellValue %} + {% endif %} + {% endfor %} + {% if cellValueHtml is empty %} + {% set cellValueHtml = ' ' %} + {% endif %} + {{ cellValueHtml|raw }} + + {% endfor %} + diff --git a/templates/components/datatable/static/formtablerow/layout.js.twig b/templates/components/datatable/static/formtablerow/layout.js.twig new file mode 100644 index 000000000..e69de29bb diff --git a/templates/components/datatable/static/layout.html.twig b/templates/components/datatable/static/layout.html.twig new file mode 100644 index 000000000..3527d1ad4 --- /dev/null +++ b/templates/components/datatable/static/layout.html.twig @@ -0,0 +1,48 @@ +{# @copyright Copyright (C) 2010-2020 Combodo SARL #} +{# @license http://opensource.org/licenses/AGPL-3.0 #} + +{% set columns = oUIBlock.GetColumns() %} + + + + {% for column in columns %} + + {% endfor %} + + + + {% for data in oUIBlock.GetData() %} + {% if data['@class'] is not empty %} + + {% else %} + + {% endif %} + {% for name,column in columns %} + + {% endfor %} + + {% endfor %} + +
{{ column.label }}
{{ cellValueHtml|raw }}
\ No newline at end of file diff --git a/templates/components/datatable/static/layout.js.twig b/templates/components/datatable/static/layout.js.twig new file mode 100644 index 000000000..8cafad35f --- /dev/null +++ b/templates/components/datatable/static/layout.js.twig @@ -0,0 +1,47 @@ +{# @copyright Copyright (C) 2010-2020 Combodo SARL #} +{# @license http://opensource.org/licenses/AGPL-3.0 #} + +{% if oUIBlock.GetOptions() is not empty %} +{% set iPageSize = oUIBlock.GetOptions()["iPageSize"] %} +{% else %} +{% set iPageSize = 10 %} +{% endif %} + +$('#{{ oUIBlock.GetId() }}').DataTable({ + language: { + processing: "{{ 'UI:Datatables:Language:Processing'|dict_s }}", + search: "{{ 'UI:Datatables:Language:Search'|dict_s }}", + lengthMenu: "{{ 'UI:Datatables:Language:LengthMenu'|dict_s }}", + zeroRecords: "{{ 'UI:Datatables:Language:ZeroRecords'|dict_s }}", + info: "{{ 'UI:Datatables:Language:Info'|dict_s }}", + infoEmpty: "", + infoFiltered: "({{ 'UI:Datatables:Language:InfoFiltered'|dict_s }})", + emptyTable: "{{ 'UI:Datatables:Language:EmptyTable'|dict_s }}", + paginate: { + first: "<<", + previous: "", + next: "", + last: ">>" + }, + aria: { + sortAscending: ": {{ 'UI:Datatables:Language:Sort:Ascending'|dict_s }}", + sortDescending: ": {{ 'UI:Datatables:Language:Sort:Descending'|dict_s }}" + } + }, + order: [], + rowId: "id", + filter: false, + {% if oUIBlock.GetData()|length <= iPageSize %} + paging: false, + {% endif %} + dom: "<'ibo-datatable-toolbar'pil>t<'ibo-datatable-toolbar'pil>", + lengthMenu: [[ {{ iPageSize }}, {{ iPageSize*2 }}, {{ iPageSize*3 }}, {{ iPageSize*4 }}, -1], [ {{ iPageSize }}, {{ iPageSize*2 }}, {{ iPageSize*3 }}, {{ iPageSize*4 }}, "{{ 'Portal:Datatables:Language:DisplayLength:All'|dict_s }}"]], + columns: [ + {% for column in oUIBlock.GetColumns() %} + { + width: "auto", + sortable: true + }, + {% endfor %} + ] +}); \ No newline at end of file diff --git a/templates/components/panel/layout.html.twig b/templates/components/panel/layout.html.twig index a7336b6e9..cf2071ddb 100644 --- a/templates/components/panel/layout.html.twig +++ b/templates/components/panel/layout.html.twig @@ -1,10 +1,10 @@ {# @copyright Copyright (C) 2010-2020 Combodo SARL #} {# @license http://opensource.org/licenses/AGPL-3.0 #} {% apply spaceless %} -
+
{% block iboPanelHeader %} -
{% block iboPanelTitle %}{{ oUIBlock.GetTitle() }}{% endblock %}
+
{% block iboPanelTitle %}{{ oUIBlock.GetTitle()|raw }}{% endblock %}
{% block iboPanelToolbar %} {% for oToolbarBlock in oUIBlock.GetToolbarBlocks() %} diff --git a/test/core/DBSearchTest.php b/test/core/DBSearchTest.php index 9127fe89f..52661517d 100644 --- a/test/core/DBSearchTest.php +++ b/test/core/DBSearchTest.php @@ -679,20 +679,21 @@ class DBSearchTest extends ItopDataTestCase $oSet = new \CMDBObjectSet($oSearch, array(), array('org_ids' => $TwoOrgIdsOnly)); static::assertEquals(4, $oSet->Count()); - $_SERVER['REQUEST_URI'] = 'FAKE_REQUEST_URI'; - $_SERVER['REQUEST_METHOD'] = 'FAKE_REQUEST_METHOD'; - $oP = new \iTopWebPage("test"); - $oBlock = new \DisplayBlock($oSet->GetFilter(), 'list', false); - $oHtml = $oBlock->GetDisplay($oP, 'package_table', array('menu' => true, 'display_limit' => false)); - $sHtml = BlockRenderer::RenderBlockTemplates($oHtml); - - $iHtmlUserRequestLineCount = substr_count($sHtml, 'GetFilter(), 'list', false); + // $oHtml = $oBlock->GetDisplay($oP, 'package_table', array('menu' => true, 'display_limit' => false)); + // $sHtml = BlockRenderer::RenderBlockTemplates($oHtml); + // + // $iHtmlUserRequestLineCount = substr_count($sHtml, '