diff --git a/.liquibase/Data_alias/basic/2021.0.2/AttributeInteger/add_attributeValidationParams.xml b/.liquibase/Data_alias/basic/2021.0.2/AttributeInteger/add_attributeValidationParams.xml
new file mode 100644
index 0000000000000000000000000000000000000000..e079628f0890a9012095934de6863a532528abe0
--- /dev/null
+++ b/.liquibase/Data_alias/basic/2021.0.2/AttributeInteger/add_attributeValidationParams.xml
@@ -0,0 +1,10 @@
+<?xml version="1.1" encoding="UTF-8" standalone="no"?>
+<databaseChangeLog xmlns="http://www.liquibase.org/xml/ns/dbchangelog"
+    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
+    xsi:schemaLocation="http://www.liquibase.org/xml/ns/dbchangelog http://www.liquibase.org/xml/ns/dbchangelog/dbchangelog-3.6.xsd">
+  <changeSet author="s.listl" id="4b3b9c3b-a0d3-4dbb-a6ab-69da2d9ae57a">
+      <addColumn tableName="AB_ATTRIBUTE">
+          <column name="VALIDATIONPARAMETERS" type="NVARCHAR(512)"/>
+      </addColumn>
+  </changeSet>
+</databaseChangeLog>
\ No newline at end of file
diff --git a/.liquibase/Data_alias/basic/2021.0.2/AttributeInteger/changelog.xml b/.liquibase/Data_alias/basic/2021.0.2/AttributeInteger/changelog.xml
new file mode 100644
index 0000000000000000000000000000000000000000..ba98ef526cbd3d91fced64c8f45af4627670362d
--- /dev/null
+++ b/.liquibase/Data_alias/basic/2021.0.2/AttributeInteger/changelog.xml
@@ -0,0 +1,7 @@
+<?xml version="1.1" encoding="UTF-8" standalone="no"?>
+<databaseChangeLog xmlns="http://www.liquibase.org/xml/ns/dbchangelog"
+    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
+    xsi:schemaLocation="http://www.liquibase.org/xml/ns/dbchangelog http://www.liquibase.org/xml/ns/dbchangelog/dbchangelog-3.6.xsd">
+    <include relativeToChangelogFile="true" file="add_attributeValidationParams.xml"/>
+    <include relativeToChangelogFile="true" file="insert_attributeTypeInteger.xml"/>
+</databaseChangeLog>
\ No newline at end of file
diff --git a/.liquibase/Data_alias/basic/2021.0.2/AttributeInteger/insert_attributeTypeInteger.xml b/.liquibase/Data_alias/basic/2021.0.2/AttributeInteger/insert_attributeTypeInteger.xml
new file mode 100644
index 0000000000000000000000000000000000000000..f8ad63956268ec21ff5bc6a169e1b5e7103c6808
--- /dev/null
+++ b/.liquibase/Data_alias/basic/2021.0.2/AttributeInteger/insert_attributeTypeInteger.xml
@@ -0,0 +1,23 @@
+<?xml version="1.1" encoding="UTF-8" standalone="no"?>
+<databaseChangeLog xmlns="http://www.liquibase.org/xml/ns/dbchangelog"
+    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
+    xsi:schemaLocation="http://www.liquibase.org/xml/ns/dbchangelog http://www.liquibase.org/xml/ns/dbchangelog/dbchangelog-3.6.xsd">
+  <changeSet author="s.listl" id="f2942a4c-96e2-4a85-81f2-153fc3f77a6c">
+        <insert tableName="AB_KEYWORD_ENTRY">
+            <column name="AB_KEYWORD_ENTRYID" value="87c376e0-8a25-4a0e-913c-885ca7fe69cc"/>
+            <column name="KEYID" value="INTEGER"/>
+            <column name="TITLE" value="Integer"/>
+            <column name="CONTAINER" value="AttributeType"/>
+            <column name="AB_KEYWORD_CATEGORY_ID" value="ad91fa20-157b-4e57-b9ad-1281e40024b3"/>
+            <column name="SORTING" valueNumeric="12"/>
+            <column name="ISACTIVE" valueNumeric="1"/>
+            <column name="ISESSENTIAL" valueNumeric="1"/>
+        </insert>
+        <insert tableName="ab_keyword_attributerelation">
+            <column name="AB_KEYWORD_ENTRY_ID" value="87c376e0-8a25-4a0e-913c-885ca7fe69cc"/>
+            <column name="AB_KEYWORD_ATTRIBUTERELATIONID" value="3e845022-cae2-4229-b199-1df03eee717e"/>
+            <column name="AB_KEYWORD_ATTRIBUTE_ID" value="307ecfc6-15c8-4ab9-8afa-e97b90a00c5f"/>
+            <column name="CHAR_VALUE" value="VAADIN:HASH"/>
+        </insert>
+  </changeSet>
+</databaseChangeLog>
\ No newline at end of file
diff --git a/.liquibase/Data_alias/basic/2021.0.2/changelog.xml b/.liquibase/Data_alias/basic/2021.0.2/changelog.xml
index 3c134bb12b911228744962c5bfea956078ea0237..62b2dafe9c72843ef097c3145e561bd27f9e3de8 100644
--- a/.liquibase/Data_alias/basic/2021.0.2/changelog.xml
+++ b/.liquibase/Data_alias/basic/2021.0.2/changelog.xml
@@ -5,4 +5,5 @@
     <include relativeToChangelogFile="true" file="alter_CompetitionExpandReasonSize.xml"/>
     <include relativeToChangelogFile="true" file="Checklists/changelog.xml"/>
     <include relativeToChangelogFile="true" file="change_SalesprojectMemberRole.xml"/>
+    <include relativeToChangelogFile="true" file="AttributeInteger/changelog.xml"/>
 </databaseChangeLog>
\ No newline at end of file
diff --git a/aliasDefinition/Data_alias/Data_alias.aod b/aliasDefinition/Data_alias/Data_alias.aod
index abce13e87dc9a3833ecaa8cc1da7dbe3f9a5fbc9..bf89272bc756faa360048b55075e1fde91cf8d9e 100644
--- a/aliasDefinition/Data_alias/Data_alias.aod
+++ b/aliasDefinition/Data_alias/Data_alias.aod
@@ -6102,6 +6102,20 @@
                 <title></title>
                 <description></description>
               </entityFieldDb>
+              <entityFieldDb>
+                <name>VALIDATIONPARAMETERS</name>
+                <dbName></dbName>
+                <primaryKey v="false" />
+                <columnType v="12" />
+                <size v="512" />
+                <scale v="0" />
+                <notNull v="false" />
+                <isUnique v="false" />
+                <index v="false" />
+                <documentation></documentation>
+                <title></title>
+                <description></description>
+              </entityFieldDb>
             </entityFields>
           </entityDb>
           <entityDb>
diff --git a/entity/AttributeRelation_entity/AttributeRelation_entity.aod b/entity/AttributeRelation_entity/AttributeRelation_entity.aod
index 6d8fcd2fb05d3259a0b339733d9eaa4b9d5a69e2..05c0bf6f0734e4ea7c8fbbd41efd67172e936955 100644
--- a/entity/AttributeRelation_entity/AttributeRelation_entity.aod
+++ b/entity/AttributeRelation_entity/AttributeRelation_entity.aod
@@ -130,6 +130,7 @@
       <stateProcess>%aditoprj%/entity/AttributeRelation_entity/entityfields/value/stateProcess.js</stateProcess>
       <valueProcess>%aditoprj%/entity/AttributeRelation_entity/entityfields/value/valueProcess.js</valueProcess>
       <displayValueProcess>%aditoprj%/entity/AttributeRelation_entity/entityfields/value/displayValueProcess.js</displayValueProcess>
+      <onValidation>%aditoprj%/entity/AttributeRelation_entity/entityfields/value/onValidation.js</onValidation>
     </entityField>
     <entityConsumer>
       <name>SpecificAttribute</name>
@@ -316,6 +317,7 @@
       <mandatoryProcess>%aditoprj%/entity/AttributeRelation_entity/entityfields/value_lookup/mandatoryProcess.js</mandatoryProcess>
       <stateProcess>%aditoprj%/entity/AttributeRelation_entity/entityfields/value_lookup/stateProcess.js</stateProcess>
       <displayValueProcess>%aditoprj%/entity/AttributeRelation_entity/entityfields/value_lookup/displayValueProcess.js</displayValueProcess>
+      <onValidation>%aditoprj%/entity/AttributeRelation_entity/entityfields/value_lookup/onValidation.js</onValidation>
     </entityField>
     <entityConsumer>
       <name>Objects</name>
@@ -389,6 +391,10 @@
         </entityParameter>
       </children>
     </entityProvider>
+    <entityField>
+      <name>VALIDATIONPARAMETERS</name>
+      <valueProcess>%aditoprj%/entity/AttributeRelation_entity/entityfields/validationparameters/valueProcess.js</valueProcess>
+    </entityField>
   </entityFields>
   <recordContainers>
     <jDitoRecordContainer>
diff --git a/entity/AttributeRelation_entity/entityfields/validationparameters/valueProcess.js b/entity/AttributeRelation_entity/entityfields/validationparameters/valueProcess.js
new file mode 100644
index 0000000000000000000000000000000000000000..fcb7189951652485e7d1a88769708b7a0ee81a5a
--- /dev/null
+++ b/entity/AttributeRelation_entity/entityfields/validationparameters/valueProcess.js
@@ -0,0 +1,12 @@
+import("system.neon");
+import("system.vars");
+import("system.result");
+import("Sql_lib");
+
+if (vars.get("$sys.recordstate") == neon.OPERATINGSTATE_NEW  || vars.get("$sys.recordstate") == neon.OPERATINGSTATE_EDIT)
+{
+    result.string(newSelect("VALIDATIONPARAMETERS")
+        .from("AB_ATTRIBUTE")
+        .where("AB_ATTRIBUTE.AB_ATTRIBUTEID", "$field.AB_ATTRIBUTE_ID")
+        .cell());
+}
\ No newline at end of file
diff --git a/entity/AttributeRelation_entity/entityfields/value/onValidation.js b/entity/AttributeRelation_entity/entityfields/value/onValidation.js
new file mode 100644
index 0000000000000000000000000000000000000000..d2f774193b62a68cfde69f37ce3ee9b740f1ac3c
--- /dev/null
+++ b/entity/AttributeRelation_entity/entityfields/value/onValidation.js
@@ -0,0 +1,12 @@
+import("Util_lib");
+import("system.result");
+import("system.vars");
+import("Attribute_lib");
+
+var attributeType = AttributeTypes.get(vars.get("$field.ATTRIBUTE_TYPE"));
+if (attributeType && !attributeType.useLookup)
+{
+    var validationResult = attributeType.validateValue(vars.get("$local.value"), Utils.parseJSON(vars.get("$field.VALIDATIONPARAMETERS")));
+    if (validationResult && validationResult !== true)
+        result.string(validationResult);
+}
\ No newline at end of file
diff --git a/entity/AttributeRelation_entity/entityfields/value_lookup/onValidation.js b/entity/AttributeRelation_entity/entityfields/value_lookup/onValidation.js
new file mode 100644
index 0000000000000000000000000000000000000000..26590927b278e6a71b75d42c2f5e9d5a4b5da26e
--- /dev/null
+++ b/entity/AttributeRelation_entity/entityfields/value_lookup/onValidation.js
@@ -0,0 +1,12 @@
+import("Util_lib");
+import("system.result");
+import("system.vars");
+import("Attribute_lib");
+
+var attributeType = AttributeTypes.get(vars.get("$field.ATTRIBUTE_TYPE"));
+if (attributeType && attributeType.useLookup)
+{
+    var validationResult = attributeType.validateValue(vars.get("$local.value"), Utils.parseJSON(vars.get("$field.VALIDATIONPARAMETERS")));
+    if (validationResult && validationResult !== true)
+        result.string(validationResult);
+}
\ No newline at end of file
diff --git a/entity/Attribute_entity/Attribute_entity.aod b/entity/Attribute_entity/Attribute_entity.aod
index 8774cf43e3a78aece8ec5c3e63aedb54c23aa218..52049d2bae1366fc8a27a2dbdc032bf69e57e1c8 100644
--- a/entity/Attribute_entity/Attribute_entity.aod
+++ b/entity/Attribute_entity/Attribute_entity.aod
@@ -497,6 +497,13 @@
         </entityParameter>
       </children>
     </entityProvider>
+    <entityField>
+      <name>VALIDATIONPARAMETERS</name>
+    </entityField>
+    <entityField>
+      <name>VALIDATIONPARAMDEFINITION</name>
+      <valueProcess>%aditoprj%/entity/Attribute_entity/entityfields/validationparamdefinition/valueProcess.js</valueProcess>
+    </entityField>
   </entityFields>
   <recordContainers>
     <jDitoRecordContainer>
@@ -514,45 +521,48 @@
           <name>UID.value</name>
         </jDitoRecordFieldMapping>
         <jDitoRecordFieldMapping>
-          <name>ATTRIBUTE_PARENT_ID.value</name>
+          <name>ATTRIBUTE_NAME.value</name>
           <isFilterable v="true" />
           <isLookupFilter v="true" />
         </jDitoRecordFieldMapping>
         <jDitoRecordFieldMapping>
-          <name>ATTRIBUTE_NAME.value</name>
-          <isFilterable v="true" />
-          <isLookupFilter v="true" />
+          <name>FULL_ATTRIBUTE_NAME.value</name>
         </jDitoRecordFieldMapping>
         <jDitoRecordFieldMapping>
-          <name>ATTRIBUTE_ACTIVE.value</name>
+          <name>ATTRIBUTE_PARENT_ID.value</name>
           <isFilterable v="true" />
           <isLookupFilter v="true" />
         </jDitoRecordFieldMapping>
         <jDitoRecordFieldMapping>
-          <name>DROPDOWNDEFINITION.value</name>
+          <name>ATTRIBUTE_PARENT_ID.displayValue</name>
         </jDitoRecordFieldMapping>
         <jDitoRecordFieldMapping>
-          <name>DROPDOWNFILTER.value</name>
+          <name>ATTRIBUTE_TYPE.value</name>
+          <isFilterable v="true" />
+          <isLookupFilter v="true" />
         </jDitoRecordFieldMapping>
         <jDitoRecordFieldMapping>
-          <name>SORTING.value</name>
+          <name>ATTRIBUTE_TYPE.displayValue</name>
         </jDitoRecordFieldMapping>
         <jDitoRecordFieldMapping>
-          <name>ATTRIBUTE_TYPE.value</name>
+          <name>ATTRIBUTE_ACTIVE.value</name>
           <isFilterable v="true" />
           <isLookupFilter v="true" />
         </jDitoRecordFieldMapping>
         <jDitoRecordFieldMapping>
-          <name>ATTRIBUTE_TYPE.displayValue</name>
+          <name>SORTING.value</name>
         </jDitoRecordFieldMapping>
         <jDitoRecordFieldMapping>
-          <name>USAGELIST.value</name>
+          <name>DROPDOWNDEFINITION.value</name>
         </jDitoRecordFieldMapping>
         <jDitoRecordFieldMapping>
-          <name>ATTRIBUTE_PARENT_ID.displayValue</name>
+          <name>DROPDOWNFILTER.value</name>
         </jDitoRecordFieldMapping>
         <jDitoRecordFieldMapping>
-          <name>FULL_ATTRIBUTE_NAME.value</name>
+          <name>VALIDATIONPARAMETERS.value</name>
+        </jDitoRecordFieldMapping>
+        <jDitoRecordFieldMapping>
+          <name>USAGELIST.value</name>
         </jDitoRecordFieldMapping>
         <jDitoRecordFieldMapping>
           <name>USAGE_FILTER.value</name>
diff --git a/entity/Attribute_entity/entityfields/dropdowndefinition/dropDownProcess.js b/entity/Attribute_entity/entityfields/dropdowndefinition/dropDownProcess.js
index 1888819bd70e3347596c0f1929902ce80c0f4158..3feaa53b20db9a340c13e426cdf5160973f28801 100644
--- a/entity/Attribute_entity/entityfields/dropdowndefinition/dropDownProcess.js
+++ b/entity/Attribute_entity/entityfields/dropdowndefinition/dropDownProcess.js
@@ -1,13 +1,13 @@
-import("system.translate");
-import("Context_lib");
 import("system.vars");
 import("system.result");
-import("Keyword_lib");
 import("Attribute_lib");
 
-var res = [];
-var type = vars.get("$field.ATTRIBUTE_TYPE").trim();
-var getDropDownFn = type && AttributeTypes[type] && AttributeTypes[type].getDropDownDefinitions;
-if (getDropDownFn instanceof Function)
-    res = getDropDownFn.call();
-result.object(res);
+var attributeType = AttributeTypes.get(vars.get("$field.ATTRIBUTE_TYPE"));
+if (attributeType)
+{
+    result.object(attributeType.getDropDownDefinitions() || []);
+}
+else
+{
+    result.object([]);
+}
\ No newline at end of file
diff --git a/entity/Attribute_entity/entityfields/dropdowndefinition/placeholderProcess.js b/entity/Attribute_entity/entityfields/dropdowndefinition/placeholderProcess.js
index ba378a119d9b896e9edabdbdabb7df1f743a4816..4b02b2b5799d21ea78a2b0b42d227e8b0186db1c 100644
--- a/entity/Attribute_entity/entityfields/dropdowndefinition/placeholderProcess.js
+++ b/entity/Attribute_entity/entityfields/dropdowndefinition/placeholderProcess.js
@@ -3,4 +3,6 @@ import("system.vars");
 import("system.result");
 import("Attribute_lib");
 
-result.string(translate.text(AttributeTypeUtil.getDropDownDefinitionTitle(vars.get("$field.ATTRIBUTE_TYPE"))));
\ No newline at end of file
+var type = AttributeTypes.get(vars.get("$field.ATTRIBUTE_TYPE"));
+var title = type ? type.dropDownDefinitionTitle : "";
+result.string(translate.text(title));
\ No newline at end of file
diff --git a/entity/Attribute_entity/entityfields/dropdowndefinition/stateProcess.js b/entity/Attribute_entity/entityfields/dropdowndefinition/stateProcess.js
index 7d544038317ed502315a9d964113718794496241..48588357052016d30ff39963139424b60444a9b1 100644
--- a/entity/Attribute_entity/entityfields/dropdowndefinition/stateProcess.js
+++ b/entity/Attribute_entity/entityfields/dropdowndefinition/stateProcess.js
@@ -4,8 +4,8 @@ import("system.result");
 import("Attribute_lib");
 
 var fieldState = neon.COMPONENTSTATE_INVISIBLE;
-var type = vars.get("$field.ATTRIBUTE_TYPE").trim();
-if (type && AttributeTypes[type] && AttributeTypes[type].getDropDownDefinitions)
+var type = AttributeTypes.get(vars.get("$field.ATTRIBUTE_TYPE"));
+if (type && type.dropDownDefinitionTitle)
 {
     if (AttributeUtil.hasRelations(vars.get("$field.UID")))
         fieldState = neon.COMPONENTSTATE_READONLY;
diff --git a/entity/Attribute_entity/entityfields/dropdowndefinition/titleProcess.js b/entity/Attribute_entity/entityfields/dropdowndefinition/titleProcess.js
index ba378a119d9b896e9edabdbdabb7df1f743a4816..4b02b2b5799d21ea78a2b0b42d227e8b0186db1c 100644
--- a/entity/Attribute_entity/entityfields/dropdowndefinition/titleProcess.js
+++ b/entity/Attribute_entity/entityfields/dropdowndefinition/titleProcess.js
@@ -3,4 +3,6 @@ import("system.vars");
 import("system.result");
 import("Attribute_lib");
 
-result.string(translate.text(AttributeTypeUtil.getDropDownDefinitionTitle(vars.get("$field.ATTRIBUTE_TYPE"))));
\ No newline at end of file
+var type = AttributeTypes.get(vars.get("$field.ATTRIBUTE_TYPE"));
+var title = type ? type.dropDownDefinitionTitle : "";
+result.string(translate.text(title));
\ No newline at end of file
diff --git a/entity/Attribute_entity/entityfields/dropdowndefinition/valueProcess.js b/entity/Attribute_entity/entityfields/dropdowndefinition/valueProcess.js
index d44ff329df79b2541e8b7c2bba9a878ce20df523..80e6eff178818d34458de8c511c2faed15ba277d 100644
--- a/entity/Attribute_entity/entityfields/dropdowndefinition/valueProcess.js
+++ b/entity/Attribute_entity/entityfields/dropdowndefinition/valueProcess.js
@@ -2,6 +2,6 @@ import("system.result");
 import("system.vars");
 import("Attribute_lib");
 
-var type = vars.get("$field.ATTRIBUTE_TYPE").trim();
-if (type && AttributeTypes[type] && typeof(AttributeTypes[type].getDropDownDefinitions) !== "function")
+var type = AttributeTypes.get(vars.get("$field.ATTRIBUTE_TYPE"));
+if (type && !type.getDropDownDefinitions())
     result.string("");
\ No newline at end of file
diff --git a/entity/Attribute_entity/entityfields/validationparamdefinition/valueProcess.js b/entity/Attribute_entity/entityfields/validationparamdefinition/valueProcess.js
new file mode 100644
index 0000000000000000000000000000000000000000..ebeed24531f076642a60756848fc2bb12d258348
--- /dev/null
+++ b/entity/Attribute_entity/entityfields/validationparamdefinition/valueProcess.js
@@ -0,0 +1,22 @@
+import("system.neon");
+import("Util_lib");
+import("system.vars");
+import("system.result");
+import("Attribute_lib");
+
+var recordstate = vars.get("$sys.recordstate");
+if (recordstate == neon.OPERATINGSTATE_NEW || recordstate == neon.OPERATINGSTATE_EDIT)
+{
+    var attributeType = AttributeTypes.get(vars.get("$field.ATTRIBUTE_TYPE"));
+    var validationParamDefinition = attributeType ? attributeType.getValidationParameters() : null;
+    if (recordstate == neon.OPERATINGSTATE_EDIT && !Utils.isNullOrEmpty(validationParamDefinition) && vars.get("$this.value") == null)
+    {
+        var parameterValues = Utils.parseJSON(vars.get("$field.VALIDATIONPARAMETERS")) || {};
+        validationParamDefinition.forEach(function (param)
+        {
+            if (param.id in parameterValues && param.value == null)
+                param.value = parameterValues[param.id];
+        });
+    }
+    result.string(validationParamDefinition ? JSON.stringify(validationParamDefinition) : "");
+}
\ No newline at end of file
diff --git a/entity/Attribute_entity/recordcontainers/jdito/contentProcess.js b/entity/Attribute_entity/recordcontainers/jdito/contentProcess.js
index 6248fae571eb24e8cd7bc79fb5b35fe45df58a6d..9f76fa5aee84c3a16afea2c3efe3900b03cf64c1 100644
--- a/entity/Attribute_entity/recordcontainers/jdito/contentProcess.js
+++ b/entity/Attribute_entity/recordcontainers/jdito/contentProcess.js
@@ -14,8 +14,8 @@ var childId = vars.get("$param.ChildId_param");
 var childType = vars.get("$param.ChildType_param");
 
 var objectType = vars.get("$param.ObjectType_param");
-var filteredIds = vars.get("$param.FilteredAttributeIds_param") ? JSON.parse(vars.getString("$param.FilteredAttributeIds_param")) : null
-var attributeCountObj = vars.get("$param.AttributeCount_param") ? JSON.parse(vars.getString("$param.AttributeCount_param")) : null;
+var filteredIds = Utils.parseJSON(vars.get("$param.FilteredAttributeIds_param"));
+var attributeCountObj = Utils.parseJSON(vars.get("$param.AttributeCount_param"));
 var displaySimpleName = Utils.toBoolean(vars.get("$param.DisplaySimpleName_param"));
 
 var themeObjectRowId = vars.get("$param.ThemeObjectRowId_param");
@@ -45,12 +45,12 @@ else if (objectType)  //if there's an objectType, it comes from the AttributeRel
 {
     translateName = true;
 
-    var ids = AttributeUtil.getPossibleAttributes(objectType, true, filteredIds, attributeCountObj);
+    var attributeIds = AttributeUtil.getPossibleAttributes(objectType, true, filteredIds, attributeCountObj);
 
-    if (Utils.isEmpty(ids))
+    if (Utils.isEmpty(attributeIds))
         condition.noResult();
     else
-        condition.and("AB_ATTRIBUTE.AB_ATTRIBUTEID", ids, SqlBuilder.IN())
+        condition.and("AB_ATTRIBUTE.AB_ATTRIBUTEID", attributeIds, SqlBuilder.IN())
             .and("AB_ATTRIBUTE.ATTRIBUTE_TYPE", AttributeTypes.THEME(), !themeObjectRowId ? SqlBuilder.NOT_EQUAL() : undefined);
 }
 else if (parentId)
@@ -90,134 +90,143 @@ var filterCondition = new FilterSqlTranslator(vars.get("$local.filter"), "AB_ATT
     .getSqlCondition();
 condition.andIfSet(filterCondition);
 
-var usages;
-if (fetchUsages) //this query is only necessary in Attribute, not in AttributeRelation
-{                        
-    var usageTbl = newSelect("AB_ATTRIBUTE_ID, OBJECT_TYPE")
-        .from("AB_ATTRIBUTEUSAGE")
-        .join("AB_ATTRIBUTE", "AB_ATTRIBUTEUSAGE.AB_ATTRIBUTE_ID = AB_ATTRIBUTE.AB_ATTRIBUTEID")
-        .whereIfSet(condition)
-        .table();
-    usages = {};
-    for (let i = 0, l = usageTbl.length; i < l; i++)
-    {
-        let attrId = usageTbl[i][0];
-        if (attrId in usages)
-            usages[attrId].push(usageTbl[i][1]);
-        else
-            usages[attrId] = [usageTbl[i][1]];
-    }
-}
+var usageLoader = new AttributeUsageLoader();
+if (fetchUsages)
+    usageLoader.fetchUsages(condition);
+
+var attributes = newSelect([
+        "AB_ATTRIBUTEID", 
+        "ATTRIBUTE_PARENT_ID", 
+        "ATTRIBUTE_NAME", 
+        "ATTRIBUTE_ACTIVE", 
+        "DROPDOWNDEFINITION", 
+        "DROPDOWNFILTER", 
+        "VALIDATIONPARAMETERS",
+        "SORTING", 
+        "ATTRIBUTE_TYPE",
+        KeywordUtils.getResolvedTitleSqlPart($KeywordRegistry.attributeType(), "ATTRIBUTE_TYPE")
+    ])
+    .from("AB_ATTRIBUTE")
+    .whereIfSet(condition)
+    .orderBy(["ATTRIBUTE_PARENT_ID", "SORTING", "ATTRIBUTE_NAME"])
+    .table();
 
-var attributes = newSelect(["AB_ATTRIBUTEID, ATTRIBUTE_PARENT_ID, ATTRIBUTE_NAME, ATTRIBUTE_ACTIVE, DROPDOWNDEFINITION, DROPDOWNFILTER, SORTING, ATTRIBUTE_TYPE",
-                            KeywordUtils.getResolvedTitleSqlPart($KeywordRegistry.attributeType(), "ATTRIBUTE_TYPE"), //3
-                            "'', '', ''"])
-                    .from("AB_ATTRIBUTE")
-                    .whereIfSet(condition)
-                    .orderBy("ATTRIBUTE_PARENT_ID, SORTING, ATTRIBUTE_NAME")
-                    .table();
+var nameResolver = new AttributeNameResolver();
+if (attributes.length !== 0)
+    nameResolver.fetchNames(translateName);
 
-var allNames = newSelect("AB_ATTRIBUTEID, ATTRIBUTE_PARENT_ID, ATTRIBUTE_NAME")
-    .from("AB_ATTRIBUTE")
-    .table(Utils.isEmpty(attributes));
-    
-var attrNameData = {};
-for (let i = 0, l = allNames.length; i < l; i++)
+var attributesById = new Map();
+attributes.forEach(function ([attributeId, parentId, simpleName, isActive, 
+    dropDownDefinition, dropDownFilter, validationParameters, sorting, type, typeName])
 {
-    attrNameData[allNames[i][0]] = [
-        allNames[i][1], 
-        translateName 
-            ? translate.text(allNames[i][2])
-            : allNames[i][2]
-    ];
-}
-var nameCache = {};
+    attributesById.set(attributeId, [
+        attributeId,
+        simpleName,
+        displaySimpleName ? nameResolver.getSimpleName(attributeId) : nameResolver.getFullName(attributeId),
+        parentId,
+        nameResolver.getFullName(parentId),
+        type,
+        typeName,
+        isActive,
+        sorting,
+        dropDownDefinition,
+        dropDownFilter,
+        validationParameters,
+        type != AttributeTypes.COMBOVALUE() ? usageLoader.getUsageSummary(attributeId) : "",
+        "usageFilter",
+        false
+    ]);
+});
+
+var resultTable = [];
+do {
+    var oldSize = resultTable.length;
+    attributesById.forEach(function (row, id)
+    {   
+        var parentId = row[3];
+        //rows that are already in the result array are removed from the attributesById Map, so if the parentId is in that Map,
+        //the parent has not been added yet
+        if (!parentId || !attributesById.has(parentId))
+        {
+            resultTable.push(row);
+            attributesById["delete"](id);
+        }
+    });
+} while (oldSize != resultTable.length); //stops the loop when no new items were added so that recursive relations between attributes don't cause an infinite loop
 
-var res = _buildAttributeTable(attributes, usages);
-result.object(res);
+result.object(resultTable);
 
-//sorts the records in a way that a tree can be built and adds values
-function _buildAttributeTable (pAttributes, pUsages) 
+
+function AttributeNameResolver ()
 {
-    var rows = {};
-    var allIds = {};
-    
-    //fills the allIds object, the object is used for checking if a parent exists in the array
-    for (let i = 0, l = pAttributes.length; i < l; i++)
-        allIds[pAttributes[i][0]] = true;
-    
-    var arrayIndex = 0;
-    
-    do {
-        var oldIndex = arrayIndex;
-        pAttributes.forEach(function (row)
-        {   
-            //item will be added if the id is not already in the object and
-            //the parent is already added (or the parent is not in the array)
-            if (!(row[0] in this) && (row[1] in this || !allIds[row[1]]))
-                this[row[0]] = {
-                    data : row,
-                    index : arrayIndex++
-                };
-        }, rows);
-    } while (oldIndex != arrayIndex); //stops the loop when no new items were added so that recursive relations between attributes don't cause an infinite loop
-    
-    var sortedArray = new Array(Object.keys(rows).length);
-    for (let i in rows)
+    this.cache = new Map();
+    this.nameData = new Map();
+    this.fetchNames = function (pTranslateNames)
     {
-        let rowData = rows[i].data;
-        if (pUsages && rowData[7].trim() != AttributeTypes.COMBOVALUE() && i in pUsages)
+        var allNames = newSelect(["AB_ATTRIBUTEID", "ATTRIBUTE_PARENT_ID", "ATTRIBUTE_NAME"])
+            .from("AB_ATTRIBUTE")
+            .table();
+        allNames.forEach(function ([attributeId, parentId, attributeName])
         {
-            rowData[9] = pUsages[i].map(function (usage)
-            {
-                return ContextUtils.getTitle(usage, true);
-            }).join(", ");
-        }
-        rowData[10] = _getFullName(rowData[1]); //parent full name
-        rowData[11] = _getFullName(rowData[0], displaySimpleName);
-        rowData[12] = "dummy";
-        rowData[13] = false;
-        sortedArray[rows[i].index] = rowData;
+            this.nameData.set(attributeId, {
+                parentId: parentId,
+                name: pTranslateNames ? translate.text(attributeName) : attributeName
+            });
+        }, this);
+    }
+    this.getSimpleName = function (pAttributeId)
+    {
+        if (this.nameData.has(pAttributeId))
+            return this.nameData.get(pAttributeId).name;
+        return "";
     }
-    
-    return sortedArray;
-    
-
-    /**
-     * builds the full attribute name from the pre-loaded parent names and adds all parent names
-     * if required
-     */
-    function _getFullName (pAttributeId, pSimpleName)
+    this.getFullName = function (pAttributeId)
     {
-        if (!pAttributeId)
+        if (!pAttributeId || !this.nameData.has(pAttributeId)) //if the id is not in this.nameData, it does not exist
             return "";
-        var attrId = pAttributeId;
-        var fullName = [];
+        if (this.cache.has(pAttributeId))
+            return this.cache.get(pAttributeId);
         
-        while (attrId)
+        var attributeData = this.nameData.get(pAttributeId);
+        var fullName = attributeData.name;
+        if (attributeData.parentId)
         {
-            let name = null;
-            if (attrId in nameCache)
-            {
-                name = nameCache[attrId];
-                attrId = null;
-            }
-            else if (attrId in attrNameData)
-            {
-                name = attrNameData[attrId][1]; 
-                attrId = attrNameData[attrId][0]; //next parent
-            }
-            else
-                attrId = null;
-            if (name)
-                fullName.unshift(name);
-            if (pSimpleName)
-                break;
+            var parentName = this.getFullName(attributeData.parentId);
+            if (parentName)
+                fullName = parentName + " / " + fullName;
         }
-        
-        fullName = fullName.join(" / ");
-        nameCache[pAttributeId] = fullName;
-
+        this.cache.set(pAttributeId, fullName);
         return fullName;
     }
 }
+
+function AttributeUsageLoader ()
+{
+    this.usageData = new Map();
+    this.fetchUsages = function (pCondition)
+    {
+        var usages = newSelect(["AB_ATTRIBUTE_ID", "OBJECT_TYPE"])
+            .from("AB_ATTRIBUTEUSAGE")
+            .join("AB_ATTRIBUTE", "AB_ATTRIBUTEUSAGE.AB_ATTRIBUTE_ID = AB_ATTRIBUTE.AB_ATTRIBUTEID")
+            .whereIfSet(pCondition)
+            .table();
+        usages.forEach(function ([attributeId, objectType])
+        {
+            if (this.usageData.has(attributeId))
+                this.usageData.get(attributeId).push(objectType);
+            else
+                this.usageData.set(attributeId, [objectType]);
+        }, this);
+    }
+    this.getUsages = function (pAttributeId)
+    {
+        return this.usageData.get(pAttributeId) || [];
+    }
+    this.getUsageSummary = function (pAttributeId)
+    {
+        return this.getUsages(pAttributeId).map(function (usage)
+        {
+            return ContextUtils.getTitle(usage, true);
+        }).join(", ");
+    }
+}
\ No newline at end of file
diff --git a/entity/Attribute_entity/recordcontainers/jdito/onInsert.js b/entity/Attribute_entity/recordcontainers/jdito/onInsert.js
index 0299b3290de6d826d7c40ba7e93f7653950fe572..0d1ffc2558febe310bac1d9c23f2959b0c2ccb50 100644
--- a/entity/Attribute_entity/recordcontainers/jdito/onInsert.js
+++ b/entity/Attribute_entity/recordcontainers/jdito/onInsert.js
@@ -15,14 +15,15 @@ if (!sorting && parentId)
 }
 
 new SqlBuilder().insertFields({
-    "AB_ATTRIBUTEID" : rowdata["UID.value"],
-    "ATTRIBUTE_PARENT_ID" : parentId,
-    "DROPDOWNDEFINITION" : rowdata["DROPDOWNDEFINITION.value"],
-    "ATTRIBUTE_ACTIVE" : rowdata["ATTRIBUTE_ACTIVE.value"],
-    "ATTRIBUTE_NAME" : rowdata["ATTRIBUTE_NAME.value"],
-    "ATTRIBUTE_TYPE" : rowdata["ATTRIBUTE_TYPE.value"],
-    "DROPDOWNFILTER" : rowdata["DROPDOWNFILTER.value"],
-    "SORTING" : sorting
+    "AB_ATTRIBUTEID": rowdata["UID.value"],
+    "ATTRIBUTE_PARENT_ID": parentId,
+    "DROPDOWNDEFINITION": rowdata["DROPDOWNDEFINITION.value"],
+    "ATTRIBUTE_ACTIVE": rowdata["ATTRIBUTE_ACTIVE.value"],
+    "ATTRIBUTE_NAME": rowdata["ATTRIBUTE_NAME.value"],
+    "ATTRIBUTE_TYPE": rowdata["ATTRIBUTE_TYPE.value"],
+    "DROPDOWNFILTER": rowdata["DROPDOWNFILTER.value"],
+    "VALIDATIONPARAMETERS": rowdata["VALIDATIONPARAMETERS.value"],
+    "SORTING": sorting
 }, "AB_ATTRIBUTE");
 
 if (rowdata["ATTRIBUTE_PARENT_ID.value"] && rowdata["ATTRIBUTE_TYPE.value"] !== AttributeTypes.COMBOVALUE() && vars.get("$param.GetOnlyFirstLevelChildren_param"))
diff --git a/entity/Attribute_entity/recordcontainers/jdito/onUpdate.js b/entity/Attribute_entity/recordcontainers/jdito/onUpdate.js
index b3012b5a5b5ff44e63a53a1dbd5c0df4b797764b..878f822d4543074e43e071e1746020aeb3fdd99c 100644
--- a/entity/Attribute_entity/recordcontainers/jdito/onUpdate.js
+++ b/entity/Attribute_entity/recordcontainers/jdito/onUpdate.js
@@ -10,6 +10,7 @@ var dbFields = {
     "ATTRIBUTE_NAME.value" : "ATTRIBUTE_NAME",
     "ATTRIBUTE_TYPE.value": "ATTRIBUTE_TYPE",
     "DROPDOWNFILTER.value": "DROPDOWNFILTER",
+    "VALIDATIONPARAMETERS.value": "VALIDATIONPARAMETERS",
     "SORTING.value" : "SORTING"
 };
 var fieldValues = {};
diff --git a/language/_____LANGUAGE_EXTRA/_____LANGUAGE_EXTRA.aod b/language/_____LANGUAGE_EXTRA/_____LANGUAGE_EXTRA.aod
index 5460db416ce6208b991345fe8be25ecebfe0bd2a..73bbfa9c8373665c9a4dc34691bfbc6f40f7909e 100644
--- a/language/_____LANGUAGE_EXTRA/_____LANGUAGE_EXTRA.aod
+++ b/language/_____LANGUAGE_EXTRA/_____LANGUAGE_EXTRA.aod
@@ -8139,6 +8139,45 @@
     <entry>
       <key>On site</key>
     </entry>
+    <entry>
+      <key>Event Data</key>
+    </entry>
+    <entry>
+      <key>On Site</key>
+    </entry>
+    <entry>
+      <key>Event Type</key>
+    </entry>
+    <entry>
+      <key>New child product</key>
+    </entry>
+    <entry>
+      <key>Communication: Link</key>
+    </entry>
+    <entry>
+      <key>Event Begin</key>
+    </entry>
+    <entry>
+      <key>Communication: Phone</key>
+    </entry>
+    <entry>
+      <key>Event</key>
+    </entry>
+    <entry>
+      <key>other</key>
+    </entry>
+    <entry>
+      <key>Communication: Mail</key>
+    </entry>
+    <entry>
+      <key>Vacation</key>
+    </entry>
+    <entry>
+      <key>Event End</key>
+    </entry>
+    <entry>
+      <key>On site</key>
+    </entry>
     <entry>
       <key>Price could not be determined</key>
     </entry>
diff --git a/language/_____LANGUAGE_de/_____LANGUAGE_de.aod b/language/_____LANGUAGE_de/_____LANGUAGE_de.aod
index 29f3d1134367471ac937b3f7dd14e3f82e2a7bed..0d380fbc994ee3a378e4c1f8e934cb8b05e1e97a 100644
--- a/language/_____LANGUAGE_de/_____LANGUAGE_de.aod
+++ b/language/_____LANGUAGE_de/_____LANGUAGE_de.aod
@@ -10,10 +10,18 @@
       <key>Event End</key>
       <value>Veranstaltungs Ende</value>
     </entry>
+    <entry>
+      <key>Value is too big, the maximum is %0</key>
+      <value>Wert ist zu groß, das Maximum ist %0</value>
+    </entry>
     <entry>
       <key>The max participants count can not be equal or less then 0</key>
       <value>Die maximale Teilnehmerzahl muss größer 0 sein!</value>
     </entry>
+    <entry>
+      <key>Value is too small, the minimum is %0</key>
+      <value>Wert ist zu klein, das Minimum ist %0</value>
+    </entry>
     <entry>
       <key>Object not found</key>
       <value>Objekt nicht gefunden
@@ -31,6 +39,10 @@
       <key>EML files can't be edited here. You can download, edit and reupload the template to change the content.</key>
       <value>Das Bearbeiten von EML-Dateien ist hier nicht möglich. Um den Inhalt zu ändern, können Sie die Vorlage herunterladen, bearbeiten und erneut hochladen.</value>
     </entry>
+    <entry>
+      <key>Value must be an integer</key>
+      <value>Wert muss eine Ganzzahl sein</value>
+    </entry>
     <entry>
       <key>Checklist entries</key>
       <value>Checklisteneinträge</value>
@@ -8999,6 +9011,10 @@ Bitte Datumseingabe prüfen</value>
       <key>Lead Imports</key>
       <value>Leadimporte</value>
     </entry>
+    <entry>
+      <key>Invalid value</key>
+      <value>Ungültiger Wert</value>
+    </entry>
     <entry>
       <key>Caimpaignsteps</key>
       <value>Kampagnenschritte</value>
diff --git a/language/_____LANGUAGE_en/_____LANGUAGE_en.aod b/language/_____LANGUAGE_en/_____LANGUAGE_en.aod
index 468e0fee73d0f2e3b3263cb589c418e5846951ff..083f0a7d6f388c4ef2561e165d9100bf6f025e90 100644
--- a/language/_____LANGUAGE_en/_____LANGUAGE_en.aod
+++ b/language/_____LANGUAGE_en/_____LANGUAGE_en.aod
@@ -8220,6 +8220,24 @@
     <entry>
       <key>Vacation</key>
     </entry>
+    <entry>
+      <key>On Site</key>
+    </entry>
+    <entry>
+      <key>New child product</key>
+    </entry>
+    <entry>
+      <key>Communication: Link</key>
+    </entry>
+    <entry>
+      <key>Communication: Phone</key>
+    </entry>
+    <entry>
+      <key>Communication: Mail</key>
+    </entry>
+    <entry>
+      <key>Vacation</key>
+    </entry>
     <entry>
       <key>Price could not be determined</key>
     </entry>
diff --git a/neonContext/Attribute/Attribute.aod b/neonContext/Attribute/Attribute.aod
index 749cae00feb54558e374c68bcbe98aa78a8c1877..46d59cc597f0e66fc980eb493a14384ea1434195 100644
--- a/neonContext/Attribute/Attribute.aod
+++ b/neonContext/Attribute/Attribute.aod
@@ -38,5 +38,8 @@
       <name>1cf7d11d-d593-4518-b7aa-aca1a9a2fb8a</name>
       <view>AttributeList_view</view>
     </neonViewReference>
+    <neonViewReference>
+      <name>ba95c103-7e26-404d-868b-ed47521bb3bd</name>
+    </neonViewReference>
   </references>
 </neonContext>
diff --git a/neonView/AttributeEdit_view/AttributeEdit_view.aod b/neonView/AttributeEdit_view/AttributeEdit_view.aod
index b044f7e623aeced5b67fa99ae4e45ec094e42e05..f579448f01e705bfc04adc4d4191199f94dddc93 100644
--- a/neonView/AttributeEdit_view/AttributeEdit_view.aod
+++ b/neonView/AttributeEdit_view/AttributeEdit_view.aod
@@ -41,6 +41,12 @@
         </entityFieldLink>
       </fields>
     </genericViewTemplate>
+    <dynamicFormViewTemplate>
+      <name>ValidationParameters</name>
+      <formDefinition>VALIDATIONPARAMDEFINITION</formDefinition>
+      <formResult>VALIDATIONPARAMETERS</formResult>
+      <editMode v="true" />
+    </dynamicFormViewTemplate>
     <neonViewReference>
       <name>8387ef27-9565-400f-a0d5-ef1d2019b722</name>
       <entityField>AttributeUsages</entityField>
diff --git a/process/Attribute_lib/process.js b/process/Attribute_lib/process.js
index e305be09e07503324dabc68aa704ea278037e92a..b6f4f0380639ea5cce991ad0f9fc62065b3c8cf5 100644
--- a/process/Attribute_lib/process.js
+++ b/process/Attribute_lib/process.js
@@ -806,19 +806,8 @@ AttributeRelationUtils.countAttributeRelations = function (pRowId, pObjectType,
  * This Object is only for the general definition of attribute types and for getting
  * data about every type, anything that has to do with a specific attribute (= the function requires an attribute id)
  * should be done in AttributeUtils.
- * The required values and methods for each type are:
  * 
- * toString = function that should return a unique name
- * contentType = the value that is returned in the contentType process for the attribute
- * databaseField = the database field that holds values of attributes with the type
- * 
- * optional:
- * getViewValue = function that gets the display value for a given value
- * getDropDownDefinitions = function that returns an array of possible values
- *          for DROPDOWNDEFINITION
- * singleSelection = if true, the maximal usage count is always 1
- * 
- * The display name is controlled by the keyword 'AttributeType'
+ * Every AttributeType also needs a keyword entry for the keyword container 'AttributeType'
  */
 function AttributeTypes () {}
 
@@ -834,14 +823,152 @@ AttributeTypes.VOID = function () {return "VOID";}
 AttributeTypes.MEMO = function () {return "MEMO";}
 AttributeTypes.OBJECTSELECTION = function () {return "OBJECTSELECTION";}
 AttributeTypes.THEME = function () {return "THEME";}
+AttributeTypes.INTEGER = function () {return "INTEGER";}
+
+/**
+ * Object for representing a single AttributeType. Note: The entries in AttributeTypes are not actually of this type, so "instanceof" will not
+ * work for them. The default properties of an attribute type are still initialized by calling this constructor, to make at least code
+ * completion possible.
+ */
+function AttributeType ()
+{
+    /**
+     * Returns a String representation of the AttributeType
+     * 
+     * @return {String} String that matches the keyId of the AttributeType Keyword
+     */
+    this.toString = function () {return this();};
+    /**
+     * ContentType of the value field
+     */
+    this.contentType = null;
+    /**
+     * Database field that should hold the value
+     */
+    this.databaseField = null;
+    /**
+     * If set to true, max usage count is always 1 (optional)
+     */
+    this.singleSelection = false;
+    /**
+     * Array of attribute types that can have this attribute as parent (optional)
+     */
+    this.possibleChildren = null;
+    /**
+     * Name of the dropdowndefinition field (optional)
+     */
+    this.dropDownDefinitionTitle = "";
+    /**
+     * Use the lookup field
+     */
+    this.useLookup = false;
+    /**
+     * Function that returns an sub-sql to resolve the attribute display value (optional)
+     * 
+     * @param {Object} pAttributeData   Object with the attribute properties
+     * @return {String} sql expression that resolves the display value
+     */
+    this.getDisplayValueSql = function (pAttributeData)
+    {
+        return this.databaseField;
+    }
+    /**
+     * Function to resolve the display value (optional)
+     * 
+     * @param {String} pValue   raw value
+     * @return {String} the display value
+     */
+    this.getViewValue = function (pValue)
+    {
+        return pValue;
+    }
+    /**
+     * Function to load all possible values for "dropdowndefinition" (optional)
+     * 
+     * @return {Array|null} array that can be used in the dropDownProcess (if defined)
+     */
+    this.getDropDownDefinitions = function () 
+    {
+        return null;
+    }
+    /**
+     * Function to define validation parameters that can be set for an attribute (optional)
+     * 
+     * @return {Object[]} dynamicForm definition for the parameter fields
+     */
+    this.getValidationParameters = function () {},
+    /**
+     * Function to validate the entered attribute value (optional)
+     * 
+     * @param {String} pValue               the value to be validated
+     * @param {Object} pValidationParams    validation parameters defined for the attribute
+     * @return {String|Boolean} A validation message string if the validation failed, true if the value is valid
+     */
+    this.validateValue = function (pValue, pValidationParams)
+    {
+        return true;
+    }
+}
+
+{   //block for encapsulation
+    for (let typeName in AttributeTypes)
+    {
+        AttributeType.call(AttributeTypes[typeName]);
+    }
+}
+
+//the "get" function is defined like this so it does not show up in for..in loops (enumerable: false)
+Object.defineProperty(AttributeTypes, "get", {
+    enumerable: false, 
+    writable: true
+});
+/**
+ * Get the AttributeType object with the given name. 
+ * 
+ * @param {String} pAttributeTypeName name of the attribute type
+ * @return {AttributeType} the attribute type, or null if the given name was invalid
+ */
+AttributeTypes.get = function (pAttributeTypeName)
+{
+    if (!pAttributeTypeName)
+        return null;
+    return AttributeTypes[pAttributeTypeName.toString().trim()] || null;
+}
+
+/*** In the following section, custom properties are defined for every AttributeType ***/
 
 Object.assign(AttributeTypes.TEXT, {
-    toString: function () {return this();},
     contentType: "TEXT",
-    databaseField: "CHAR_VALUE"
+    databaseField: "CHAR_VALUE",
+    getValidationParameters: function ()
+    {
+        return [{
+            id: "regExp",
+            name: translate.text("Regular expression"),
+            contentType: "TEXT",
+            isReadable: true,
+            isWritable: true,
+            isRequired: false,
+            value: null
+        }];
+    },
+    validateValue: function (pValue, pValidationParams)
+    {
+        if (pValidationParams && pValidationParams.regExp)
+        {
+            var regExParts = pValidationParams.regExp.match(new RegExp('^/(.*?)/([gimy]*)$'));
+            var regEx;
+            if (regExParts)
+                regEx = new RegExp(regExParts[1], regExParts[2]);
+            else
+                regEx = new RegExp(pValidationParams.regExp);
+            if (!regEx.test(pValue))
+                return translate.text("Invalid value");
+        }
+        return true;
+    }
 });
 Object.assign(AttributeTypes.DATE, {
-    toString: function () {return this();},
     contentType: "DATE", 
     databaseField: "DATE_VALUE",
     getViewValue: function (pValue)
@@ -850,12 +977,42 @@ Object.assign(AttributeTypes.DATE, {
     }
 });
 Object.assign(AttributeTypes.NUMBER, {
-    toString: function () {return this();},
     contentType: "NUMBER", 
-    databaseField: "NUMBER_VALUE"
+    databaseField: "NUMBER_VALUE",
+    getValidationParameters: function ()
+    {
+        return [{
+            id: "minValue",
+            name: translate.text("Minimum"),
+            contentType: "NUMBER",
+            isReadable: true,
+            isWritable: true,
+            isRequired: false,
+            value: null
+        },{
+            id: "maxValue",
+            name: translate.text("Maximum"),
+            contentType: "NUMBER",
+            isReadable: true,
+            isWritable: true,
+            isRequired: false,
+            value: null
+        }];
+    },
+    validateValue: function (pValue, pValidationParams)
+    {
+        if (pValidationParams)
+        {   
+            pValue = Number(pValue);
+            if (!Utils.isNullOrEmptyString(pValidationParams.minValue) && pValue < pValidationParams.minValue)
+                return translate.withArguments("Value is too small, the minimum is %0", [pValidationParams.minValue]);
+            if (!Utils.isNullOrEmptyString(pValidationParams.maxValue) && pValue > pValidationParams.maxValue)
+                return translate.withArguments("Value is too big, the maximum is %0", [pValidationParams.maxValue]);
+        }
+        return true;
+    }
 });
 Object.assign(AttributeTypes.BOOLEAN, {
-    toString: function () {return this();},
     contentType: "BOOLEAN", 
     databaseField: "INT_VALUE",
     singleSelection: true,
@@ -871,7 +1028,6 @@ Object.assign(AttributeTypes.BOOLEAN, {
     }
 });
 Object.assign(AttributeTypes.COMBO, {
-    toString: function () {return this();},
     contentType: "UNKNOWN",
     databaseField: "ID_VALUE",
     possibleChildren: [AttributeTypes.COMBOVALUE()],
@@ -901,15 +1057,7 @@ Object.assign(AttributeTypes.COMBO, {
         return viewValue ? translate.text(viewValue) : viewValue;
     }
 });
-Object.assign(AttributeTypes.COMBOVALUE, {
-    toString: function () {return this();},
-    contentType: null, 
-    databaseField: null
-});
 Object.assign(AttributeTypes.GROUP, {
-    toString: function () {return this();},
-    contentType: null, 
-    databaseField: null,
     possibleChildren: [
         AttributeTypes.GROUP(), 
         AttributeTypes.TEXT(), 
@@ -925,7 +1073,6 @@ Object.assign(AttributeTypes.GROUP, {
     ]
 });
 Object.assign(AttributeTypes.KEYWORD, {
-    toString: function () {return this();},
     contentType: "UNKNOWN", 
     databaseField: "ID_VALUE", 
     getDisplayValueSql: function (pAttributeData)
@@ -947,21 +1094,17 @@ Object.assign(AttributeTypes.KEYWORD, {
     }
 });
 Object.assign(AttributeTypes.VOID, {
-    toString: function () {return this();},
-    contentType: null,
-    databaseField: null,
     possibleChildren: [AttributeTypes.VOID()],
     singleSelection: true
 });
 Object.assign(AttributeTypes.MEMO, {
-    toString: function () {return this();},
     contentType: "LONG_TEXT", 
     databaseField: "CHAR_VALUE"
 });
 Object.assign(AttributeTypes.OBJECTSELECTION, {
-    toString: function () {return this();},
     contentType: "UNKNOWN",
     databaseField: "ID_VALUE",
+    useLookup: true,
     getViewValue: function (pValue, pModule)
         {
             if (pValue)
@@ -1028,11 +1171,48 @@ Object.assign(AttributeTypes.OBJECTSELECTION, {
     ])
 });
 Object.assign(AttributeTypes.THEME, {
-    toString: function () {return this();},
     contentType: "LONG_TEXT",
     databaseField: "CHAR_VALUE",
     possibleChildren: [AttributeTypes.THEME()]
 });
+Object.assign(AttributeTypes.INTEGER, {
+    contentType: "NUMBER",
+    databaseField: "INT_VALUE",
+    getValidationParameters: function ()
+    {
+        return [{
+            id: "minValue",
+            name: translate.text("Minimum"),
+            contentType: "NUMBER",
+            isReadable: true,
+            isWritable: true,
+            isRequired: false,
+            value: null
+        },{
+            id: "maxValue",
+            name: translate.text("Maximum"),
+            contentType: "NUMBER",
+            isReadable: true,
+            isWritable: true,
+            isRequired: false,
+            value: null
+        }];
+    },
+    validateValue: function (pValue, pValidationParams)
+    {
+        if (!Utils.isInteger(pValue))
+            return translate.text("Value must be an integer");
+        if (pValidationParams)
+        {   
+            pValue = Number(pValue);
+            if (!Utils.isNullOrEmptyString(pValidationParams.minValue) && pValue < pValidationParams.minValue)
+                return translate.withArguments("Value is too small, the minimum is %0", [pValidationParams.minValue]);
+            if (!Utils.isNullOrEmptyString(pValidationParams.maxValue) && pValue > pValidationParams.maxValue)
+                return translate.withArguments("Value is too big, the maximum is %0", [pValidationParams.maxValue]);
+        }
+        return true;
+    }
+})
 
 //reference for compatibility with old name
 var $AttributeTypes = AttributeTypes;
@@ -1050,7 +1230,8 @@ function AttributeTypeUtil () {}
  */
 AttributeTypeUtil.getContentType = function (pAttributeType)
 {
-    return AttributeTypeUtil._getProperty(pAttributeType, "contentType");
+    var type = AttributeTypes.get(pAttributeType);
+    return type ? type.contentType : null;
 }
 
 /**
@@ -1064,7 +1245,8 @@ AttributeTypeUtil.getContentType = function (pAttributeType)
  */
 AttributeTypeUtil.isGroupType = function (pAttributeType)
 {
-    return !Utils.isNullOrEmpty(AttributeTypeUtil.getPossibleChildren(pAttributeType));
+    var type = AttributeTypes.get(pAttributeType);
+    return type && !Utils.isNullOrEmpty(type.possibleChildren);
 }
 
 /**
@@ -1081,7 +1263,8 @@ AttributeTypeUtil.isGroupType = function (pAttributeType)
  */
 AttributeTypeUtil.getDatabaseField = function (pAttributeType)
 {
-    return AttributeTypeUtil._getProperty(pAttributeType, "databaseField");
+    var type = AttributeTypes.get(pAttributeType);
+    return type ? type.databaseField : null;
 }
 
 /**
@@ -1091,12 +1274,13 @@ AttributeTypeUtil.getDatabaseField = function (pAttributeType)
  *                                              The attribute type (use the values <br>
  *                                              of the AttributeTypes object, e. g.<br>
  *                                              AttributeTypes.TEXT)<br>
- * @return {String[]}                           <p>
- *                                              The possible children types.<br>
+ * @return {String[]|null}                      <p>
+ *                                              The possible children types, can be null.<br>
  */
 AttributeTypeUtil.getPossibleChildren = function (pAttributeType)
 {
-    return AttributeTypeUtil._getProperty(pAttributeType, "possibleChildren");
+    var type = AttributeTypes.get(pAttributeType);
+    return type ? type.possibleChildren : null;
 }
 
 /**
@@ -1107,12 +1291,13 @@ AttributeTypeUtil.getPossibleChildren = function (pAttributeType)
  *                                              The attribute type (use the values<br>
  *                                              of the AttributeTypes object, e. g.<br>
  *                                              AttributeTypes.TEXT)<br>
- * @return {String[]}                           <p>
- *                                              .<br>
+ * @return {Boolean}                            <p>
+ *                                              if the attribute can only be used once<br>
  */
 AttributeTypeUtil.isSingleSelection = function (pAttributeType)
 {
-    return AttributeTypeUtil._getProperty(pAttributeType, "singleSelection", false);
+    var type = AttributeTypes.get(pAttributeType);
+    return type ? type.singleSelection : null;
 }
 
 /**
@@ -1127,23 +1312,26 @@ AttributeTypeUtil.isSingleSelection = function (pAttributeType)
  */
 AttributeTypeUtil.getDropDownDefinitionTitle = function (pAttributeType)
 {
-    return AttributeTypeUtil._getProperty(pAttributeType, "dropDownDefinitionTitle", "");
+    var type = AttributeTypes.get(pAttributeType);
+    return type ? type.dropDownDefinitionTitle : "";
 }
 
+/**
+ * Returns a function to resolve the displayValue depending on the attribute type.
+ * 
+ * @param {String} pAttributeType               <p>
+ *                                              The attribute type (use the values<br>
+ *                                              of the AttributeTypes object, e. g.<br>
+ *                                              AttributeTypes.TEXT)<br>
+ * @return {Function}                           <p>
+ *                                              A function that resolves the displayValue or null if the type is invalid<br>
+ */
 AttributeTypeUtil.getDisplayValueSqlFn = function (pAttributeType)
 {
-    if (!pAttributeType)
-        return null;
-    
-    pAttributeType = pAttributeType.trim();
-    var attributeType = AttributeTypes[pAttributeType];
+    var attributeType = AttributeTypes.get(pAttributeType);
     if (!attributeType)
         return null;
-    var displayValueSqlFn = attributeType.getDisplayValueSql || function () 
-    {
-        return this.databaseField;
-    };
-    return displayValueSqlFn.bind(attributeType);
+    return attributeType.getDisplayValueSql.bind(attributeType);
 }
 
 /**
@@ -1152,14 +1340,15 @@ AttributeTypeUtil.getDisplayValueSqlFn = function (pAttributeType)
  * 
  * @param {String} pAttributeType               <p>
  *                                              The attribute type which shall be comapred.
- * @return {String}                             <p>
+ * @return {Boolean}                            <p>
  *                                              Returns true, if the given attribute type is equal<br>
  *                                              with the attribute string "OBJECTSELECTION" and <br>
  *                                              false, if not.<br>                                             
  */
 AttributeTypeUtil.useLookup = function (pAttributeType)
 {
-    return pAttributeType.trim() == AttributeTypes.OBJECTSELECTION();
+    var type = AttributeTypes.get(pAttributeType);
+    return type ? type.useLookup : false;
 }
 
 /**
@@ -1183,37 +1372,6 @@ AttributeTypeUtil.getGroupTypes = function (pChildType)
     return groupTypes;
 }
 
-/**
- * Returns the property matching to the property name.<br>
- * 
- * @param {String} pAttributeType           <p>
- *                                          The corresponding attribute type to the<br>
- *                                          property.
- * @param {String} pPropertyName            <p>
- *                                          The property name of the property.
- * @param {String} pDefaultValue            <p>
- *                                          Description.
- * @return                                  <p>
- *                                          Returns the property or null, when pAttributeType<br>
- *                                          isn't filled or the given attribute type isn't in<br>
- *                                          AttributeTypes. Otherwise the pDefaultValue,<br>
- *                                          will be returned, case if it isn't undefined or null.<br>
- */
-AttributeTypeUtil._getProperty = function (pAttributeType, pPropertyName, pDefaultValue)
-{
-    if (!pAttributeType)
-        return null;
-    
-    pAttributeType = pAttributeType.trim();
-    if (pAttributeType in AttributeTypes)
-        if (pPropertyName in AttributeTypes[pAttributeType])
-            return AttributeTypes[pAttributeType][pPropertyName];
-        else
-            return pDefaultValue === undefined ? null : pDefaultValue;
-        
-    return null;
-}
-
 /**
  * If the given attribute type is a <br>
  * valid type and it has a getViewValue<br>
@@ -1231,9 +1389,8 @@ AttributeTypeUtil._getProperty = function (pAttributeType, pPropertyName, pDefau
  */
 AttributeTypeUtil.getAttributeViewValue = function (pAttributeType, pValue, pKeyword)
 {
-    if (pAttributeType in AttributeTypes && AttributeTypes[pAttributeType].getViewValue)
-        return AttributeTypes[pAttributeType].getViewValue(pValue, pKeyword);
-    return pValue;
+    var type = AttributeTypes.get(pAttributeType);
+    return type ? type.getViewValue(pValue, pKeyword) : pValue;
 }
 
 /**
diff --git a/process/Util_lib/process.js b/process/Util_lib/process.js
index 10f2b630c4c66072046df07d331f3c77dd451493..8ea572643943a2724f7a323e207122ff530957de 100644
--- a/process/Util_lib/process.js
+++ b/process/Util_lib/process.js
@@ -52,6 +52,33 @@ Utils.isNullOrEmpty = function (pObject)
     return pObject === null || pObject === undefined || Utils.isEmpty(pObject);
 }
 
+/**
+ * Checks if the given value is null, undefined or "". This is useful for "has no value" checks where 0 and false
+ * should count as explicit values. 
+ * 
+ * @param {String} pValue the value to check
+ * @return {Boolean} true if the value is null, undefined or ""
+ */
+Utils.isNullOrEmptyString = function (pValue)
+{
+    return pValue === null || pValue === undefined || pValue === "";
+}
+
+/**
+ * Checks if the given value is not null, undefined or "". This is useful for "has value" checks where 0 and false
+ * should count as explicit values.
+ * 
+ * @param {String} pValue the value to check
+ * @return {Boolean} true if the value is null, undefined or ""
+ * @example 
+ * 
+ * var filteredArray = myArray.filter(Utils.isNotNullOrEmptyString);
+ */
+Utils.isNotNullOrEmptyString = function (pValue)
+{
+    return pValue !== null && pValue !== undefined && pValue !== "";
+}
+
 /**
  * Creates a deep copy of the given object. Also works with arrays, maps and sets.
  *