Pārlūkot izejas kodu

User editable dashboards... implementation in progress

git-svn-id: http://svn.code.sf.net/p/itop/code/trunk@1999 a333f486-631f-4898-b8df-5754b55c2be0
dflaven 13 gadi atpakaļ
vecāks
revīzija
1d02491131

+ 34 - 1
application/dashboard.class.inc.php

@@ -1,7 +1,26 @@
 <?php
+// Copyright (C) 2012 Combodo SARL
+//
+//   This program is free software; you can redistribute it and/or modify
+//   it under the terms of the GNU General Public License as published by
+//   the Free Software Foundation; version 3 of the License.
+//
+//   This program is distributed in the hope that it will be useful,
+//   but WITHOUT ANY WARRANTY; without even the implied warranty of
+//   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+//   GNU General Public License for more details.
+//
+//   You should have received a copy of the GNU General Public License
+//   along with this program; if not, write to the Free Software
+//   Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
+
 require_once(APPROOT.'application/dashboardlayout.class.inc.php');
 require_once(APPROOT.'application/dashlet.class.inc.php');
 
+/**
+ * A user editable dashboard page
+ *
+ */
 abstract class Dashboard
 {
 	protected $sTitle;
@@ -84,6 +103,7 @@ abstract class Dashboard
 		if (!$bEditMode)
 		{
 			$oPage->add_linked_script('../js/dashlet.js');
+			$oPage->add_linked_script('../js/property_field.js');
 		}
 	}
 	
@@ -149,16 +169,29 @@ abstract class Dashboard
 		$oPage->add('<div id="dashlet_properties" style="text-align:center">');
 		foreach($this->aDashlets as $oDashlet)
 		{
-			$oDashlet->RenderProperties($oPage);
+			$sId = $oDashlet->GetID();
+			$sClass = get_class($oDashlet);
+			
+			$oPage->add('<div class="dashlet_properties" id="dashlet_properties_'.$sId.'" style="display:none">');
+			$oForm = $oDashlet->GetForm($oPage);
+			$this->SetFormParams($oForm);
+			$oForm->RenderAsPropertySheet($oPage);		
+			$oPage->add('</div>');
 		}
 		$oPage->add('</div>');
 
 		$oPage->add('</div>');
 	}
+	
+	abstract protected function SetFormParams($oForm);
 }
 
 class RuntimeDashboard extends Dashboard
 {
+	protected function SetFormParams($oForm)
+	{
+		$oForm->SetSubmitParams(utils::GetAbsoluteUrlAppRoot().'pages/ajax.render.php', array('operation' => 'update_dashlet_property'));		
+	}
 	public function Save()
 	{
 		

+ 87 - 60
application/dashlet.class.inc.php

@@ -1,11 +1,37 @@
 <?php
+// Copyright (C) 2012 Combodo SARL
+//
+//   This program is free software; you can redistribute it and/or modify
+//   it under the terms of the GNU General Public License as published by
+//   the Free Software Foundation; version 3 of the License.
+//
+//   This program is distributed in the hope that it will be useful,
+//   but WITHOUT ANY WARRANTY; without even the implied warranty of
+//   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+//   GNU General Public License for more details.
+//
+//   You should have received a copy of the GNU General Public License
+//   along with this program; if not, write to the Free Software
+//   Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
+
+require_once(APPROOT.'application/forms.class.inc.php');
+
+/**
+ * Base class for all 'dashlets' (i.e. widgets to be inserted into a dashboard)
+ *
+ */
 abstract class Dashlet
 {
 	protected $sId;
+	protected $bRedrawNeeded;
+	protected $bFormRedrawNeeded;
+	
 	
 	public function __construct($sId)
 	{
 		$this->sId = $sId;
+		$this->bRedrawNeeded = true; // By default: redraw each time a property changes
+		$this->bFormRedrawNeeded = false; // By default: no need to redraw the form (independent fields)
 	}
 	
 	public function FromDOMNode($oDOMNode)
@@ -53,30 +79,27 @@ EOF
 	}
 	
 	abstract public function Render($oPage, $bEditMode = false, $aExtraParams = array());
+		
+	abstract public function GetPropertiesFields(DesignerForm $oForm);
 	
-	public function RenderProperties($oPage)
+	public function ToXml(DOMNode $oContainerNode)
 	{
-		$sId = $this->GetID();
-		$sClass = get_class($this);
-		$oPage->add('<div class="dashlet_properties" id="dashlet_properties_'.$sId.'" style="display:none">');
-		$oPage->add("<p>Properties for $sClass / $sId</p>");
-		$oPage->add('</div>');
+		
 	}
-	
-	
-	public function ToXml(DOMNode $oContainerNode)
+
+	public function Update($aValues, $aUpdatedFields)
 	{
 		
 	}
 	
-	public function GetForm()
+	public function IsRedrawNeeded()
 	{
-		
+		return $this->bRedrawNeeded;
 	}
 	
-	public function OnFieldUpdate($aParams, $sUpdatedFieldCode)
+	public function IsFormRedrawNeeded()
 	{
-		
+		return $this->bFormRedrawNeeded;
 	}
 	
 	static public function GetInfo()
@@ -87,55 +110,79 @@ EOF
 			'description' => '',
 		);
 	}
+	
+	public function GetForm($oPage, $bReturnHTML = false)
+	{
+		$oForm = new DesignerForm();
+		$oForm->SetPrefix("dashlet_". $this->GetID());
+		$oForm->SetParamsContainer('params');
+		
+		$this->GetPropertiesFields($oForm);
+		
+		$oDashletClassField = new DesignerHiddenField('dashlet_class', '', get_class($this));
+		$oForm->AddField($oDashletClassField);
+		
+		$oDashletIdField = new DesignerHiddenField('dashlet_id', '', $this->GetID());
+		$oForm->AddField($oDashletIdField);
+		
+		return $oForm;
+	}
 }
 
 class DashletHelloWorld extends Dashlet
 {
+	protected $sText;
+	
 	public function __construct($sId)
 	{
 		parent::__construct($sId);
+		$this->sText = 'Hello World';
 	}
 	
 	public function FromDOMNode($oDOMNode)
 	{
-		
+		//$this->sText = 'Hello World!';
 	}
 	
 	public function FromXml($sXml)
 	{
-		
+		//$this->sText = 'Hello World!';
 	}
 	
 	public function FromParams($aParams)
 	{
-		
+		$this->sText = $aParams['text'];
 	}
 	
-	public function Render($oPage, $bEditMode = false, $aExtraParams = array())
+	public function Update($aValues, $aUpdatedFields)
 	{
-		$oPage->add('<div style="text-align:center; line-height:5em" class="dashlet-content"><span>Hello World!</span></div>');
+		foreach($aUpdatedFields as $sProp)
+		{
+			switch($sProp)
+			{
+				case 'text':
+				$this->sText = $aValues['text'];
+				break;
+			}
+		}
 	}
 	
-	public function ToXml(DOMNode $oContainerNode)
+	public function Render($oPage, $bEditMode = false, $aExtraParams = array())
 	{
-		$oNewNodeNode = $oContainerNode->ownerDocument->createElement('hello_world', 'test');
-		$oContainerNode->appendChild($oNewNodeNode);
+		$oPage->add('<div style="text-align:center; line-height:5em" class="dashlet-content"><span>'.$this->sText.'</span></div>');
 	}
-	
-	public function GetForm()
+	public function GetPropertiesFields(DesignerForm $oForm)
 	{
-		
+		$oField = new DesignerTextField('text', 'Text', $this->sText);
+		$oForm->AddField($oField);
 	}
 	
-	public function OnFieldUpdate($aParams, $sUpdatedFieldCode)
+	public function ToXml(DOMNode $oContainerNode)
 	{
-		return array(
-			'status_ok' => true,
-			'redraw' => false,
-			'fields' => array(),
-		);
+		$oNewNodeNode = $oContainerNode->ownerDocument->createElement('hello_world', 'test');
+		$oContainerNode->appendChild($oNewNodeNode);
 	}
-	
+
 	static public function GetInfo()
 	{
 		return array(
@@ -174,26 +221,16 @@ class DashletFakeBarChart extends Dashlet
 		$oPage->add('<div style="text-align:center" class="dashlet-content"><div>Fake Bar Chart</div><divp><img src="../images/fake-bar-chart.png"/></div></div>');
 	}
 	
-	public function ToXml(DOMNode $oContainerNode)
+	public function GetPropertiesFields(DesignerForm $oForm, $oDashlet = null)
 	{
-		$oNewNodeNode = $oContainerNode->ownerDocument->createElement('fake_bar_chart', 'test');
-		$oContainerNode->appendChild($oNewNodeNode);
-	}
-	
-	public function GetForm()
-	{
-		
 	}
 	
-	public function OnFieldUpdate($aParams, $sUpdatedFieldCode)
+	public function ToXml(DOMNode $oContainerNode)
 	{
-		return array(
-			'status_ok' => true,
-			'redraw' => false,
-			'fields' => array(),
-		);
+		$oNewNodeNode = $oContainerNode->ownerDocument->createElement('fake_bar_chart', 'test');
+		$oContainerNode->appendChild($oNewNodeNode);
 	}
-	
+
 	static public function GetInfo()
 	{
 		return array(
@@ -232,24 +269,14 @@ class DashletFakePieChart extends Dashlet
 		$oPage->add('<div style="text-align:center" class="dashlet-content"><div>Fake Pie Chart</div><div><img src="../images/fake-pie-chart.png"/></div></div>');
 	}
 	
-	public function ToXml(DOMNode $oContainerNode)
-	{
-		$oNewNodeNode = $oContainerNode->ownerDocument->createElement('fake_pie_chart', 'test');
-		$oContainerNode->appendChild($oNewNodeNode);
-	}
-	
-	public function GetForm()
+	public function GetPropertiesFields(DesignerForm $oForm, $oDashlet = null)
 	{
-		
 	}
 	
-	public function OnFieldUpdate($aParams, $sUpdatedFieldCode)
+	public function ToXml(DOMNode $oContainerNode)
 	{
-		return array(
-			'status_ok' => true,
-			'redraw' => false,
-			'fields' => array(),
-		);
+		$oNewNodeNode = $oContainerNode->ownerDocument->createElement('fake_pie_chart', 'test');
+		$oContainerNode->appendChild($oNewNodeNode);
 	}
 	
 	static public function GetInfo()

+ 984 - 0
application/forms.class.inc.php

@@ -0,0 +1,984 @@
+<?php
+// Copyright (C) 2012 Combodo SARL
+//
+//   This program is free software; you can redistribute it and/or modify
+//   it under the terms of the GNU General Public License as published by
+//   the Free Software Foundation; version 3 of the License.
+//
+//   This program is distributed in the hope that it will be useful,
+//   but WITHOUT ANY WARRANTY; without even the implied warranty of
+//   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+//   GNU General Public License for more details.
+//
+//   You should have received a copy of the GNU General Public License
+//   along with this program; if not, write to the Free Software
+//   Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
+
+/**
+ * Helper class to build interactive forms to be used either in stand-alone
+ * modal dialog or in "property-sheet" panes.
+ *
+ */
+class DesignerForm
+{
+	protected $aFieldSets;
+	protected $sCurrentFieldSet;
+	protected $sScript;
+	protected $sReadyScript;
+	protected $sFormId;
+	protected $sFormPrefix;
+	protected $sParamsContainer;
+	protected $oParentForm;
+	protected $aSubmitParams;
+	protected $sSubmitTo;
+	protected $bReadOnly;
+	
+	public function __construct()
+	{
+		$this->aFieldSets = array();
+		$this->sCurrentFieldSet = '';
+		$this->sScript = '';
+		$this->sReadyScript = '';
+		$this->sFormPrefix = '';
+		$this->sParamsContainer = '';
+		$this->sFormId = 'form_'.rand();
+		$this->oParentForm = null;
+		$this->bReadOnly = false;
+		$this->StartFieldSet($this->sCurrentFieldSet);
+	}
+	
+	public function AddField(DesignerFormField $oField)
+	{
+		if (!is_array($this->aFieldSets[$this->sCurrentFieldSet]))
+		{
+			$this->aFieldSets[$this->sCurrentFieldSet] = array();
+		}
+		$this->aFieldSets[$this->sCurrentFieldSet][] = $oField;
+		$oField->SetForm($this);
+	}
+	
+	public function StartFieldSet($sLabel)
+	{
+		$this->sCurrentFieldSet = $sLabel;
+		if (!array_key_exists($this->sCurrentFieldSet, $this->aFieldSets))
+		{
+			$this->aFieldSets[$this->sCurrentFieldSet] = array();
+		}
+	}
+	
+	public function Render($oP, $bReturnHTML = false)
+	{
+		$sReturn = '';
+		if ($this->oParentForm == null)
+		{
+			$sFormId = $this->sFormId;
+			$sReturn = '<form id="'.$sFormId.'">';
+		}
+		else
+		{
+			$sFormId = $this->oParentForm->sFormId;
+		}
+		$sHiddenFields = '';
+		foreach($this->aFieldSets as $sLabel => $aFields)
+		{
+			$aDetails = array();
+			if ($sLabel != '')
+			{
+				$sReturn .= '<fieldset>';
+				$sReturn .= '<legend>'.$sLabel.'</legend>';
+			}
+			foreach($aFields as $oField)
+			{
+				$aRow = $oField->Render($oP, $sFormId);
+				if ($oField->IsVisible())
+				{
+					$aDetails[] = array('label' => $aRow['label'], 'value' => $aRow['value']);
+				}
+				else
+				{
+					$sHiddenFields .= $aRow['value'];
+				}
+			}
+			$sReturn .= $oP->GetDetails($aDetails);
+			if ($sLabel != '')
+			{
+				$sReturn .= '</fieldset>';
+			}
+		}
+		$sReturn .= $sHiddenFields;
+		
+		if ($this->oParentForm == null)
+		{
+			$sReturn .= '</form>';
+		}
+		if($this->sScript != '')
+		{
+			$oP->add_script($this->sScript);
+		}
+		if($this->sReadyScript != '')
+		{
+			$oP->add_ready_script($this->sReadyScript);
+		}
+		if ($bReturnHTML)
+		{
+			return $sReturn;
+		}
+		else
+		{
+			$oP->add($sReturn);
+		}
+	}
+
+	public function SetSubmitParams($sSubmitToUrl, $aSubmitParams)
+	{
+		$this->sSubmitTo = $sSubmitToUrl;
+		$this->aSubmitParams = $aSubmitParams;
+	}
+	
+	public function CopySubmitParams($oParentForm)
+	{
+		$this->sSubmitTo = $oParentForm->sSubmitTo;
+		$this->aSubmitParams = $oParentForm->aSubmitParams;
+	}
+	
+	
+	public function RenderAsPropertySheet($oP, $bReturnHTML = false)
+	{
+		$sReturn = '';		
+		$sActionUrl = addslashes($this->sSubmitTo);
+		$sJSSubmitParams = json_encode($this->aSubmitParams);
+		if ($this->oParentForm == null)
+		{
+			$sFormId = $this->sFormId;
+			$sReturn = '<form id="'.$sFormId.'">';
+			$sReturn .= '<table class="prop_table">';
+			$sReturn .= '<thead><tr><th class="prop_header">Property</th><th class="prop_header">Value</th><th colspan="2" class="prop_header">&nbsp;</th></tr></thead><tbody>';
+		}
+		else
+		{
+			$sFormId = $this->oParentForm->sFormId;
+		}
+		$sHiddenFields = '';
+		foreach($this->aFieldSets as $sLabel => $aFields)
+		{
+			$aDetails = array();
+			if ($sLabel != '')
+			{
+				$sReturn .= '<tr><th colspan="4">'.$sLabel.'</th></tr>';
+			}
+
+
+			$sValidationFields = '</td><td class="prop_icon prop_apply"><span title="Apply" class="ui-icon ui-icon-circle-check"/></td><td  class="prop_icon prop_cancel"><span title="Revert" class="ui-icon ui-icon-circle-close"/></td></tr>';
+			foreach($aFields as $oField)
+			{
+				$aRow = $oField->Render($oP, $sFormId, 'property');
+				if ($oField->IsVisible())
+				{
+					$sFieldId = $this->GetFieldId($oField->GetCode());
+					$sReturn .= '<tr id="row_'.$sFieldId.'"><td class="prop_label">'.$aRow['label'].'</td><td class="prop_value">'.$aRow['value'];
+					if (!($oField instanceof DesignerFormSelectorField))
+					{
+						$sReturn .= $sValidationFields;
+					}
+					$this->AddReadyScript(
+<<<EOF
+$('#row_$sFieldId').property_field({field_id: '$sFieldId', value: '', submit_to: '$sActionUrl', submit_parameters: $sJSSubmitParams });
+EOF
+					);
+				}
+				else
+				{
+					$sHiddenFields .= $aRow['value'];
+				}
+			}
+		}
+		
+		if ($this->oParentForm == null)
+		{
+			$sFormId = $this->sFormId;
+			$sReturn .= '</tbody>';
+			$sReturn .= '</table>';
+			$sReturn .= $sHiddenFields;
+			$sReturn .= '</form>';
+			$sReturn .= '<div id="prop_submit_result"/>'; // for the return of the submit operation
+		}
+		else
+		{
+			$sReturn .= $sHiddenFields;
+		}
+		$this->AddReadyScript(
+<<<EOF
+		$('.prop_table').tableHover();
+		var idx = 0;
+		$('.prop_table tbody tr').each(function() {
+			if ((idx % 2) == 0)
+			{
+				$(this).addClass('even');
+			}
+			else
+			{
+				$(this).addClass('odd');
+			}
+			idx++;
+		});
+EOF
+		);
+		
+		if($this->sScript != '')
+		{
+			$oP->add_script($this->sScript);
+		}
+		if($this->sReadyScript != '')
+		{
+			$oP->add_ready_script($this->sReadyScript);
+		}
+		if ($bReturnHTML)
+		{
+			return $sReturn;
+		}
+		else
+		{
+			$oP->add($sReturn);
+		}
+	}	
+	public function RenderAsDialog($oPage, $sDialogId, $sDialogTitle, $iDialogWidth, $sOkButtonLabel)
+	{
+		$sDialogTitle = addslashes($sDialogTitle);
+		$sOkButtonLabel = addslashes($sOkButtonLabel);
+		$sCancelButtonLabel = 'Cancel'; //TODO: localize
+		$oPage->add("<div id=\"$sDialogId\">");
+		$this->Render($oPage);
+		$oPage->add('</div>');
+		
+		$oPage->add_ready_script(
+<<<EOF
+$('#$sDialogId').dialog({
+		height: 'auto',
+		width: 500,
+		modal: true,
+		title: '$sDialogTitle',
+		buttons: [
+		{ text: "$sOkButtonLabel", click: function() {
+			var oForm = $(this).parents('.ui-dialog :first').find('form');
+			oForm.submit();
+		} },
+		{ text: "$sCancelButtonLabel", click: function() { KillAllMenus(); $(this).dialog( "close" ); $(this).remove(); } },
+		],
+		close: function() { KillAllMenus(); $(this).remove(); }
+	});
+	var oForm = $('#$sDialogId form');
+	var sFormId = oForm.attr('id');
+	ValidateForm(sFormId, true);
+EOF
+		);		
+	}
+	
+	public function ReadParams(&$aValues = array())
+	{
+		foreach($this->aFieldSets as $sLabel => $aFields)
+		{
+			foreach($aFields as $oField)
+			{
+				$oField->ReadParam($aValues);
+			}
+		}
+		return $aValues;
+	}
+	
+	public function SetPrefix($sPrefix)
+	{
+		$this->sFormPrefix = $sPrefix;
+	}
+	
+	public function GetPrefix()
+	{
+		return $this->sFormPrefix;
+	}
+	
+	public function SetReadOnly($bReadOnly = true)
+	{
+		$this->bReadOnly = $bReadOnly;
+	}
+	
+	public function IsReadOnly()
+	{
+		if ($this->oParentForm == null)
+		{
+			return $this->bReadOnly;
+		}
+		else
+		{
+			return $this->oParentForm->IsReadOnly();
+		}
+	}
+	
+	public function SetParamsContainer($sParamsContainer)
+	{
+		$this->sParamsContainer = $sParamsContainer;
+	}
+	
+	public function GetParamsContainer()
+	{
+		if ($this->oParentForm == null)
+		{
+			return $this->sParamsContainer;
+		}
+		else
+		{
+			return $this->oParentForm->GetParamsContainer();
+		}
+	}
+	
+	public function SetParentForm($oParentForm)
+	{
+		$this->oParentForm = $oParentForm;
+	}
+	
+	public function AddScript($sScript)
+	{
+		$this->sScript .= $sScript;
+	}
+	
+	public function AddReadyScript($sScript)
+	{
+		$this->sReadyScript .= $sScript;
+	}
+	
+	public function GetFieldId($sCode)
+	{
+		return $this->sFormPrefix.'attr_'.$sCode;
+	}
+	
+	public function GetFieldName($sCode)
+	{
+		return 'attr_'.$sCode;
+	}
+	
+	public function GetParamName($sCode)
+	{
+		return 'attr_'.$sCode;
+	}
+	
+	public function GetValidationArea($sCode)
+	{
+		return "<span style=\"display:inline-block;width:20px;\" id=\"v_{$this->sFormPrefix}attr_$sCode\"></span>";
+	}
+	public function GetAsyncActionClass()
+	{
+		return $this->sAsyncActionClass;
+	}
+}
+
+class DesignerTabularForm extends DesignerForm
+{
+	protected $aTable;
+	
+	public function __construct()
+	{
+		parent::__construct();
+		$this->aTable = array();
+	}
+	public function AddRow($aRow)
+	{
+		$this->aTable[] = $aRow;
+	}
+	
+	public function Render($oP, $bReturnHTML = false)
+	{
+		$sReturn = '';
+		if ($this->oParentForm == null)
+		{
+			$sFormId = $this->sFormId;
+			$sReturn = '<form id="'.$sFormId.'">';
+		}
+		else
+		{
+			$sFormId = $this->oParentForm->sFormId;
+		}
+		$sHiddenFields = '';
+		$sReturn .= '<table style="width:100%">';
+		foreach($this->aTable as $aRow)
+		{
+			$sReturn .= '<tr>';
+			foreach($aRow as $field)
+			{
+				if (!is_object($field))
+				{
+					// Shortcut: pass a string for a cell containing just a label
+					$sReturn .= '<td>'.$field.'</td>';
+				}
+				else
+				{
+					$field->SetForm($this);
+					$aFieldData = $field->Render($oP, $sFormId);
+					if ($field->IsVisible())
+					{
+						// put the label and value separated by a non-breaking space if needed
+						$aData = array();
+						foreach(array('label', 'value') as $sCode )
+						{
+							if ($aFieldData[$sCode] != '')
+							{
+								$aData[] = $aFieldData[$sCode];
+							}						
+						}
+						$sReturn .= '<td>'.implode('&nbsp;', $aData).'</td>';
+					}
+					else
+					{
+						$sHiddenFields .= $aRow['value'];
+					}
+				}
+			}
+			$sReturn .= '</tr>';
+		}
+		$sReturn .= '</table>';
+		
+		$sReturn .= $sHiddenFields;
+		
+		if($this->sScript != '')
+		{
+			$oP->add_script($this->sScript);
+		}
+		if($this->sReadyScript != '')
+		{
+			$oP->add_ready_script($this->sReadyScript);
+		}
+		if ($bReturnHTML)
+		{
+			return $sReturn;
+		}
+		else
+		{
+			$oP->add($sReturn);
+		}
+	}
+	
+	public function ReadParams(&$aValues = array())
+	{
+		foreach($this->aTable as $aRow)
+		{
+			foreach($aRow as $field)
+			{
+				if (is_object($field))
+				{
+					$field->SetForm($this);
+					$field->ReadParam($aValues);
+				}
+			}
+		}
+		return $aValues;
+	}
+}
+
+class DesignerFormField
+{
+	protected $sLabel;
+	protected $sCode;
+	protected $defaultValue;
+	protected $oForm;
+	protected $bMandatory;
+	protected $bReadOnly;
+	
+	public function __construct($sCode, $sLabel, $defaultValue)
+	{
+		$this->sLabel = $sLabel;
+		$this->sCode = $sCode;
+		$this->defaultValue = $defaultValue;
+		$this->bMandatory = false;
+		$this->bReadOnly = false;
+	}
+	
+	public function GetCode()
+	{
+		return $this->sCode;
+	}
+	
+	public function SetForm($oForm)
+	{
+		$this->oForm = $oForm;
+	}
+	
+
+	public function SetMandatory($bMandatory = true)
+	{
+		$this->bMandatory = $bMandatory;
+	}
+
+	public function SetReadOnly($bReadOnly = true)
+	{
+		$this->bReadOnly = $bReadOnly;
+	}
+	
+	public function IsReadOnly()
+	{
+		return ($this->oForm->IsReadOnly() || $this->bReadOnly);
+	}
+	
+	public function Render(WebPage $oP, $sFormId, $sRenderMode='dialog')
+	{
+		$sId = $this->oForm->GetFieldId($this->sCode);
+		$sName = $this->oForm->GetFieldName($this->sCode);
+		return array('label' => $this->sLabel, 'value' => "<input type=\"text\" id=\"$sId\" name=\"$sName\" value=\"".htmlentities($this->defaultValue, ENT_QUOTES, 'UTF-8')."\">");
+	}
+	
+	public function ReadParam(&$aValues)
+	{
+		if ($this->IsReadOnly())
+		{
+			$aValues[$this->sCode] = $this->defaultValue;
+		}
+		else
+		{
+			if ($this->oForm->GetParamsContainer() != '')
+			{
+				$aParams = utils::ReadParam($this->oForm->GetParamsContainer(), array(), false, 'raw_data');
+				if (array_key_exists($this->oForm->GetParamName($this->sCode), $aParams))
+				{
+					$aValues[$this->sCode] = $aParams[$this->oForm->GetParamName($this->sCode)];
+				}
+				else
+				{
+					$aValues[$this->sCode] = $this->defaultValue;
+				}
+			}
+			else
+			{
+				$aValues[$this->sCode] = utils::ReadParam($this->oForm->GetParamName($this->sCode), $this->defaultValue, false, 'raw_data');
+			}
+		}
+	}
+	
+	public function IsVisible()
+	{
+		return true;
+	}
+}
+
+class DesignerLabelField extends DesignerFormField
+{
+	protected $sDescription;
+	
+	public function __construct($sLabel, $sDescription)
+	{
+		parent::__construct('', $sLabel, '');
+		$this->sDescription = $sDescription;
+	}
+	
+	public function Render(WebPage $oP, $sFormId, $sRenderMode='dialog')
+	{
+		$sId = $this->oForm->GetFieldId($this->sCode);
+		$sName = $this->oForm->GetFieldName($this->sCode);
+		return array('label' => $this->sLabel, 'value' => $sDescription);
+	}
+	
+	public function ReadParam(&$aValues)
+	{
+	}
+	
+	public function IsVisible()
+	{
+		return true;
+	}
+}
+
+class DesignerTextField extends DesignerFormField
+{
+	protected $sValidationPattern;
+	public function __construct($sCode, $sLabel = '', $defaultValue = '')
+	{
+		parent::__construct($sCode, $sLabel, $defaultValue);
+		$this->sValidationPattern = '';
+	}
+	
+	public function SetValidationPattern($sValidationPattern)
+	{
+		$this->sValidationPattern = $sValidationPattern;
+	}
+	
+	public function Render(WebPage $oP, $sFormId, $sRenderMode='dialog')
+	{
+		$sId = $this->oForm->GetFieldId($this->sCode);
+		$sName = $this->oForm->GetFieldName($this->sCode);
+		$sValidation = $this->oForm->GetValidationArea($this->sCode);
+		$sPattern = addslashes($this->sValidationPattern);
+		$sMandatory = $this->bMandatory ? 'true' :  'false';
+		$sReadOnly = $this->IsReadOnly() ? 'readonly' :  '';
+		$oP->add_ready_script(
+<<<EOF
+$('#$sId').bind('change keyup validate', function() { ValidateWithPattern('$sId', $sMandatory, '$sPattern', '$sFormId'); } );
+{
+	var myTimer = null;
+	$('#$sId').bind('keyup', function() { clearTimeout(myTimer); myTimer = setTimeout(function() { $('#$sId').trigger('change', {} ); }, 100); });
+}
+EOF
+);
+		return array('label' => $this->sLabel, 'value' => "<input type=\"text\" id=\"$sId\" $sReadOnly name=\"$sName\" value=\"".htmlentities($this->defaultValue, ENT_QUOTES, 'UTF-8')."\">".$sValidation);
+	}
+
+	public function ReadParam(&$aValues)
+	{
+		parent::ReadParam($aValues);
+
+		if (($this->sValidationPattern != '') &&(!preg_match('/'.$this->sValidationPattern.'/', $aValues[$this->sCode])) ) 
+		{
+			$aValues[$this->sCode] = $this->defaultValue;
+		}
+	}
+}
+
+class DesignerComboField extends DesignerFormField
+{
+	protected $aAllowedValues;
+	protected $bMultipleSelection;
+	protected $bOtherChoices;
+	
+	public function __construct($sCode, $sLabel = '', $defaultValue = '')
+	{
+		parent::__construct($sCode, $sLabel, $defaultValue);
+		$this->aAllowedValues = array();
+		$this->bMultipleSelection = false;
+		$this->bOtherChoices = false;
+	}
+	
+	public function SetAllowedValues($aAllowedValues)
+	{
+		$this->aAllowedValues = $aAllowedValues;
+	}
+	
+	public function MultipleSelection($bMultipleSelection = true)
+	{
+		$this->bMultipleSelection = $bMultipleSelection;
+	}
+	
+	public function OtherChoices($bOtherChoices = true)
+	{
+		$this->bOtherChoices = $bOtherChoices;
+	}
+	
+	
+	public function Render(WebPage $oP, $sFormId, $sRenderMode='dialog')
+	{
+
+		$sId = $this->oForm->GetFieldId($this->sCode);
+		$sName = $this->oForm->GetFieldName($this->sCode);
+		$sChecked = $this->defaultValue ? 'checked' : '';
+		$sMandatory = $this->bMandatory ? 'true' :  'false';
+		$sReadOnly = $this->IsReadOnly() ? 'disabled="disabled"' :  '';
+		$sValidation = $this->oForm->GetValidationArea($this->sCode);
+		if ($this->bMultipleSelection)
+		{
+			$sHtml = "<select multiple size=\"8\"id=\"$sId\" name=\"$sName\" $sReadOnly>";
+		}
+		else
+		{
+			$sHtml = "<select id=\"$sId\" name=\"$sName\" $sReadOnly>";
+			$sHtml .= "<option value=\"\">".Dict::S('UI:SelectOne')."</option>";
+		}
+		foreach($this->aAllowedValues as $sKey => $sDisplayValue)
+		{
+			if ($this->bMultipleSelection)
+			{
+				$sSelected = in_array($sKey, $this->defaultValue) ? 'selected' : '';
+			}
+			else
+			{
+				$sSelected = ($sKey == $this->defaultValue) ? 'selected' : '';
+			}
+			// Quick and dirty: display the menu parents as a tree
+			$sHtmlValue = str_replace(' ', '&nbsp;', htmlentities($sDisplayValue, ENT_QUOTES, 'UTF-8'));
+			$sHtml .= "<option value=\"".htmlentities($sKey, ENT_QUOTES, 'UTF-8')."\" $sSelected>$sHtmlValue</option>";
+		}
+		$sHtml .= "</select>";
+		if ($this->bOtherChoices)
+		{
+			$sHtml .= '<br/><input type="checkbox" id="other_chk_'.$sId.'"><label for="other_chk_'.$sId.'">&nbsp;Other:</label>&nbsp;<input type="text" id="other_'.$sId.'" name="other_'.$sName.'" size="30"/>'; 
+		}
+		$oP->add_ready_script(
+<<<EOF
+$('#$sId').bind('change validate', function() { ValidateWithPattern('$sId', $sMandatory, '', '$sFormId'); } );
+EOF
+);
+		return array('label' => $this->sLabel, 'value' => $sHtml.$sValidation);
+	}
+}
+
+class DesignerBooleanField extends DesignerFormField
+{
+	public function __construct($sCode, $sLabel = '', $defaultValue = '')
+	{
+		parent::__construct($sCode, $sLabel, $defaultValue);
+	}
+	
+	public function Render(WebPage $oP, $sFormId, $sRenderMode='dialog')
+	{
+		$sId = $this->oForm->GetFieldId($this->sCode);
+		$sName = $this->oForm->GetFieldName($this->sCode);
+		$sValidation = $this->oForm->GetValidationArea($this->sCode);
+		$sChecked = $this->defaultValue ? 'checked' : '';
+		$sReadOnly = $this->IsReadOnly() ? 'disabled' :  ''; // readonly does not work as expected on checkboxes:
+															 // readonly prevents the user from changing the input's value not its state (checked/unchecked)
+		return array('label' => $this->sLabel, 'value' => "<input type=\"checkbox\" $sChecked $sReadOnly id=\"$sId\" name=\"$sName\" value=\"true\">".$sValidation);
+	}
+	
+	public function ReadParam(&$aValues)
+	{
+		if ($this->IsReadOnly())
+		{
+			$aValues[$this->sCode] = $this->defaultValue;
+		}
+		else
+		{
+			$sParamsContainer = $this->oForm->GetParamsContainer();
+			if ($sParamsContainer != '')
+			{
+				$aParams = utils::ReadParam($sParamsContainer, array(), false, 'raw_data');
+				if (array_key_exists($this->oForm->GetParamName($this->sCode), $aParams))
+				{
+					$sValue = $aParams[$this->oForm->GetParamName($this->sCode)];
+				}
+				else
+				{
+					$sValue = 'false';
+				}
+			}
+			else
+			{
+				$sValue = utils::ReadParam($this->oForm->GetParamName($this->sCode), 'false', false, 'raw_data');
+			}
+		}
+		$aValues[$this->sCode] = ($sValue == 'true');
+	}
+}
+
+
+class DesignerHiddenField extends DesignerFormField
+{
+	public function __construct($sCode, $sLabel = '', $defaultValue = '')
+	{
+		parent::__construct($sCode, $sLabel, $defaultValue);
+	}
+	
+	public function IsVisible()
+	{
+		return false;
+	}
+	
+	public function Render(WebPage $oP, $sFormId, $sRenderMode='dialog')
+	{
+		$sId = $this->oForm->GetFieldId($this->sCode);
+		$sName = $this->oForm->GetFieldName($this->sCode);
+		$sValidation = $this->oForm->GetValidationArea($this->sCode);
+		$sChecked = $this->defaultValue ? 'checked' : '';
+		return array('label' =>'', 'value' => "<input type=\"hidden\" id=\"$sId\" name=\"$sName\" value=\"".htmlentities($this->defaultValue, ENT_QUOTES, 'UTF-8')."\">");
+	}
+}
+
+
+class DesignerIconSelectionField extends DesignerFormField
+{
+	protected $aAllowedValues;
+	
+	public function __construct($sCode, $sLabel = '', $defaultValue = '')
+	{
+		parent::__construct($sCode, $sLabel, $defaultValue);
+	}
+	
+	public function SetAllowedValues($aAllowedValues)
+	{
+		$this->aAllowedValues = $aAllowedValues;
+	}
+	
+	public function Render(WebPage $oP, $sFormId, $sRenderMode='dialog')
+	{
+		$sId = $this->oForm->GetFieldId($this->sCode);
+		$sName = $this->oForm->GetFieldName($this->sCode);
+		$idx = 0;
+		foreach($this->aAllowedValues as $index => $aValue)
+		{
+			if ($aValue['value'] == $this->defaultValue)
+			{
+				$idx = $index;
+				break;
+			}
+		}
+		$sJSItems = json_encode($this->aAllowedValues);
+		if (!$this->IsReadOnly())
+		{
+			$oP->add_ready_script(
+<<<EOF
+	$('#$sId').icon_select({current_idx: $idx, items: $sJSItems});
+EOF
+			);
+		}
+		$sReadOnly = $this->IsReadOnly() ? 'disabled' : '';
+		return array('label' =>$this->sLabel, 'value' => "<input type=\"hidden\" id=\"$sId\" name=\"$sName\" value=\"{$this->defaultValue}\"/>");
+	}
+}
+
+class DesignerSortableField extends DesignerFormField
+{
+	protected $aAllowedValues;
+	
+	public function __construct($sCode, $sLabel = '', $defaultValue = '')
+	{
+		parent::__construct($sCode, $sLabel, $defaultValue);
+		$this->aAllowedValues = array();
+	}
+	
+	public function SetAllowedValues($aAllowedValues)
+	{
+		$this->aAllowedValues = $aAllowedValues;
+	}
+	
+	public function Render(WebPage $oP, $sFormId, $sRenderMode='dialog')
+	{
+		$bOpen = false;
+		$sId = $this->oForm->GetFieldId($this->sCode);
+		$sName = $this->oForm->GetFieldName($this->sCode);
+		$sValidation = $this->oForm->GetValidationArea($this->sCode);
+		$sHtml = "<span class=\"sort_$sId fieldslist\" id=\"sortable_$sId\">";
+		foreach($this->defaultValue as $sValue)
+		{
+			$sHtml .= "<span class=\"movable_attr\">$sValue</span>";
+		}
+		$sHtml .="</span>";
+		$sIconClass = $bOpen ? 'ui-icon-circle-triangle-s' : 'ui-icon-circle-triangle-e';
+		$sStyle = $bOpen ? '' : 'style="display:none"';
+		$sHtml .= "<div class=\"fieldspicker\"><table><tr><td><span id=\"collapse_$sId\" class=\"ui-icon $sIconClass\" style=\"opacity: 0.5\"></span>Fields</td></tr><tr><td><div $sStyle id=\"fieldsbasket_$sId\" class=\"sort_$sId fieldsbasket\">";
+		foreach($this->aAllowedValues as $sKey => $sDisplayValue)
+		{
+			$sHtml .= "<span class=\"movable_attr\">$sDisplayValue</span>";
+		}
+		$sHtml .="</div></td></tr>";
+		$sHtml .="<tr id=\"trash_icon_$sId\" $sStyle><td><span class=\"ui-icon ui-icon-trash\" style=\"opacity: 0.5\"></span>Trash</td></tr><tr id=\"trash_$sId\" $sStyle><td><div id=\"recycle_$sId\" class=\"sort_$sId fieldstrash\"></div></div></td></tr></table></div>";
+		$oP->add_ready_script(
+<<<EOF
+	$('#collapse_$sId').click(function() { $(this).toggleClass('ui-icon-circle-triangle-s').toggleClass('ui-icon-circle-triangle-e'); $('#fieldsbasket_$sId').toggle(); $('#trash_icon_$sId').toggle(); $('#trash_$sId').toggle(); } );
+	$('#fieldsbasket_$sId .movable_attr').draggable({connectToSortable: '#sortable_$sId', helper: 'clone', revert: false });
+	$('#recycle_$sId').sortable({ receive: function(event, ui) { ui.item.animate({opacity: 0.25}, { complete: function() { $(this).remove(); } });} });
+	$('#sortable_$sId').sortable({connectWith: '#recycle_$sId', forcePlaceholderSize: true});
+	$('#sortable_$sId').disableSelection();
+EOF
+		);
+		return array('label' => $this->sLabel, 'value' => $sHtml.$sValidation);
+	}
+}
+
+class DesignerFormSelectorField extends DesignerFormField
+{
+	protected $aSubForms;
+	protected $defaultRealValue; // What's stored as default value is actually the index
+	public function __construct($sCode, $sLabel = '', $defaultValue = '')
+	{
+		parent::__construct($sCode, $sLabel, 0);
+		$this->defaultRealValue = $defaultValue;
+		$this->aSubForms = array();
+	}
+	
+	public function AddSubForm($oSubForm, $sLabel, $sValue)
+	{
+		$idx = count($this->aSubForms);
+		$this->aSubForms[] = array('form' => $oSubForm, 'label' => $sLabel, 'value' => $sValue);
+		if ($sValue == $this->defaultRealValue)
+		{
+			// Store the index of the selected/default form
+			$this->defaultValue = count($this->aSubForms) - 1;
+		}
+	}
+	
+	public function Render(WebPage $oP, $sFormId, $sRenderMode='dialog')
+	{
+		$sId = $this->oForm->GetFieldId($this->sCode);
+		$sName = $this->oForm->GetFieldName($this->sCode);
+		$sReadOnly = $this->IsReadOnly() ? 'disabled="disabled"' :  '';
+		
+		$sHtml = "<select id=\"$sId\" name=\"$sName\" $sReadOnly>";
+		foreach($this->aSubForms as $sKey => $aFormData)
+		{
+			$sDisplayValue = $aFormData['label'];
+			$sSelected = ($sKey == $this->defaultValue) ? 'selected' : '';
+			$sHtml .= "<option value=\"".htmlentities($sKey, ENT_QUOTES, 'UTF-8')."\" $sSelected>".htmlentities($sDisplayValue, ENT_QUOTES, 'UTF-8')."</option>";
+		}
+		$sHtml .= "</select>";
+		
+		if ($sRenderMode == 'property')
+		{
+			$sHtml .= '</td><td class="prop_icon prop_apply"><span title="Apply" class="ui-icon ui-icon-circle-check"/></td><td  class="prop_icon prop_cancel"><span title="Revert" class="ui-icon ui-icon-circle-close"/></td></tr>';
+		}
+				
+		foreach($this->aSubForms as $sKey => $aFormData)
+		{
+			$sId = $this->oForm->GetFieldId($this->sCode);
+			$sStyle = ($sKey == $this->defaultValue) ? '' : 'style="display:none"';
+			$oSubForm = $aFormData['form'];
+			$oSubForm->SetParentForm($this->oForm);
+			$oSubForm->CopySubmitParams($this->oForm);
+			$oSubForm->SetPrefix($this->oForm->GetPrefix().$sKey.'_');
+			
+			if ($sRenderMode == 'property')
+			{
+				$sHtml .= "</tbody><tbody class=\"subform\" id=\"{$sId}_{$sKey}\" $sStyle>";
+				$sHtml .= $oSubForm->RenderAsPropertySheet($oP, true);	
+			}
+			else
+			{
+				$sHtml .= "<div class=\"subform\" id=\"{$sId}_{$sKey}\" $sStyle>";
+				$sHtml .= $oSubForm->Render($oP, true);
+				$sHtml .= "</div>";
+			}
+		}
+
+		$oP->add_ready_script(
+<<<EOF
+$('#$sId').bind('change reverted', function() { $('.subform').hide(); $('#{$sId}_'+this.value).show(); } );
+EOF
+);
+		return array('label' => $this->sLabel, 'value' => $sHtml);
+	}
+
+	public function ReadParam(&$aValues)
+	{
+		parent::ReadParam($aValues);
+		$sKey = $aValues[$this->sCode];
+		$aValues[$this->sCode] = $this->aSubForms[$sKey]['value'];
+		
+		$this->aSubForms[$sKey]['form']->SetPrefix($this->oForm->GetPrefix().$sKey.'_');
+		$this->aSubForms[$sKey]['form']->SetParentForm($this->oForm);
+		$this->aSubForms[$sKey]['form']->ReadParams($aValues);
+	}
+}
+
+class DesignerSubFormField extends DesignerFormField
+{
+	protected $oSubForm;
+	public function __construct($sLabel, $oSubForm)
+	{
+		parent::__construct('', $sLabel, '');
+		$this->oSubForm = $oSubForm;
+	}
+	
+	public function Render(WebPage $oP, $sFormId, $sRenderMode='dialog')
+	{
+		$this->oSubForm->SetParentForm($this->oForm);
+		$oSubForm->CopySubmitParams($this->oForm);
+		
+		if ($sRenderMode == 'property')
+		{
+			$sHtml = $this->oSubForm->RenderAsPropertySheet($oP, true);
+		}
+		else
+		{
+			$sHtml = $this->oSubForm->Render($oP, true);
+		}
+		return array('label' => $this->sLabel, 'value' => $sHtml);
+	}
+
+	public function ReadParam(&$aValues)
+	{	
+		$this->oSubForm->SetParentForm($this->oForm);
+		$this->oSubForm->ReadParams($aValues);
+	}
+}
+
+?>