diff --git a/aliasDefinition/Data_alias/indexsearchgroups/org/affectedIds.js b/aliasDefinition/Data_alias/indexsearchgroups/org/affectedIds.js index ae199b2018a9652efde2cbe29362e0e1f0e0b58c..59126ad58df30299bc21dcb629b3fef2b781a50e 100644 --- a/aliasDefinition/Data_alias/indexsearchgroups/org/affectedIds.js +++ b/aliasDefinition/Data_alias/indexsearchgroups/org/affectedIds.js @@ -2,12 +2,13 @@ import("system.db"); import("system.result"); import("system.vars"); import("IndexSearch_lib"); -var indexHelper, infoContainer, onUpdFn, tableName, res; +import("Sql_lib"); + +var infoContainer, onUpdFn, tableName, res; -indexHelper = new IndexsearchUtils(); tableName = vars.get("$local.table"); idValue = vars.get("$local.idvalue"); -infoContainer = indexHelper.createAffectedInfoContainer(idValue, null, vars.get("$local.action") +infoContainer = IndexsearchUtils.createAffectedInfoContainer(idValue, null, vars.get("$local.action") ,function (){return vars.get("$local.columns")} ,function (){return vars.get("$local.oldvalues")} ,function (){return vars.get("$local.values")}); @@ -21,16 +22,16 @@ switch (tableName) res = db.array(db.COLUMN, "select RELATION.RELATIONID from RELATION where RELATION.PERS_ID is null and RELATION.ORG_ID = '" + idValue + "'"); break; case "ADDRESS": - res = indexHelper.getAffectedIdValues("RELATION_ID", infoContainer, function (id){ + res = IndexsearchUtils.getAffectedIdValues("RELATION_ID", infoContainer, function (id){ return db.array(db.COLUMN, ["select ADDRESS.RELATION_ID from ADDRESS where ADDRESS.ADDRESSID = ?", [ - [id, db.getColumnTypes("ADDRESS", ["ADDRESSID"])[0]] + [id, SqlUtils.getSingleColumnType("ADDRESS", ["ADDRESSID"])] ]]); }); break; case "COMM": - res = indexHelper.getAffectedIdValues("RELATION_ID", infoContainer, function (id){ + res = IndexsearchUtils.getAffectedIdValues("RELATION_ID", infoContainer, function (id){ return db.array(db.COLUMN, ["select ADDRESS.RELATION_ID from COMM where COMMID = ?", [ - [id, db.getColumnTypes("COMM", ["COMMID"])[0]] + [id, SqlUtils.getSingleColumnType("COMM", ["COMMID"])] ]]); }); break; diff --git a/aliasDefinition/Data_alias/indexsearchgroups/pers/affectedIds.js b/aliasDefinition/Data_alias/indexsearchgroups/pers/affectedIds.js index a87cc105b4964eab1b273f808b2586b93b041a24..80d3d35764628599b1812c367844a7433711b413 100644 --- a/aliasDefinition/Data_alias/indexsearchgroups/pers/affectedIds.js +++ b/aliasDefinition/Data_alias/indexsearchgroups/pers/affectedIds.js @@ -2,12 +2,13 @@ import("system.db"); import("system.result"); import("system.vars"); import("IndexSearch_lib"); -var indexHelper, infoContainer, onUpdFn, tableName, res; +import("Sql_lib"); + +var infoContainer, onUpdFn, tableName, res; -indexHelper = new IndexsearchUtils(); tableName = vars.get("$local.table"); idValue = vars.get("$local.idvalue"); -infoContainer = indexHelper.createAffectedInfoContainer(idValue, null, vars.get("$local.action") +infoContainer = IndexsearchUtils.createAffectedInfoContainer(idValue, null, vars.get("$local.action") ,function (){return vars.get("$local.columns")} ,function (){return vars.get("$local.oldvalues")} ,function (){return vars.get("$local.values")}); @@ -24,16 +25,16 @@ switch (tableName) res = db.array(db.COLUMN, "select RELATION.RELATIONID from RELATION where RELATION.PERS_ID is not null and RELATION.ORG_ID = '" + idValue + "'"); break; case "ADDRESS": - res = indexHelper.getAffectedIdValues("RELATION_ID", infoContainer, function (id){ + res = IndexsearchUtils.getAffectedIdValues("RELATION_ID", infoContainer, function (id){ return db.array(db.COLUMN, ["select ADDRESS.RELATION_ID from ADDRESS where ADDRESS.ADDRESSID = ?", [ - [id, db.getColumnTypes("ADDRESS", ["ADDRESSID"])[0]] + [id, SqlUtils.getSingleColumnType("ADDRESS", ["ADDRESSID"])] ]]); }); break; case "COMM": - res = indexHelper.getAffectedIdValues("RELATION_ID", infoContainer, function (id){ + res = IndexsearchUtils.getAffectedIdValues("RELATION_ID", infoContainer, function (id){ return db.array(db.COLUMN, ["select ADDRESS.RELATION_ID from COMM where COMMID = ?", [ - [id, db.getColumnTypes("COMM", ["COMMID"])[0]] + [id, SqlUtils.getSingleColumnType("COMM", ["COMMID"])] ]]); }); break; diff --git a/entity/Comm_entity/entityfields/addr/onValidation.js b/entity/Comm_entity/entityfields/addr/onValidation.js index 97fccc1a01aeaa1794aae039d5bf8f8d00ddca22..f7e031675a3de8e05eb86384e2ee943f5ab9de88 100644 --- a/entity/Comm_entity/entityfields/addr/onValidation.js +++ b/entity/Comm_entity/entityfields/addr/onValidation.js @@ -7,6 +7,7 @@ import("system.mail"); import("Keyword_lib"); import("Comm_lib"); import("Util_lib"); +import("Entity_lib"); var kwdUtil = new KeywordUtils(); var kwd = kwdUtil.createKeyword("COMM.MEDIUM"); @@ -15,7 +16,7 @@ var commCategory = kwd.getPropForKey(commMedium, "contentType", true);//TODO: ma var fn = CommValidationUtil.makeValidationFn(commCategory); if (fn != null){ - var commAddr = ProcessHandlingUtil.getOnValidationValue(vars.get("$field.ADDR")); + var commAddr = ProcessHandlingUtils.getOnValidationValue(vars.get("$field.ADDR")); var additional = CommValidationUtil.getExtensionsBlueprint(); additional.countryCode = vars.get("$param.RelationsMainCountry_param");//TODO: try to use users language first and then the companies var res = fn.call(null, commAddr, additional); diff --git a/entity/Contract_entity/entityfields/contractcode/onValidation.js b/entity/Contract_entity/entityfields/contractcode/onValidation.js index e357339a3f7c17483b1c7188c54a1c547174f19a..8d2de85d894a32c92d53fd9e16cffb2b1fd153fb 100644 --- a/entity/Contract_entity/entityfields/contractcode/onValidation.js +++ b/entity/Contract_entity/entityfields/contractcode/onValidation.js @@ -3,8 +3,12 @@ import("system.result"); import("system.vars"); import("system.db"); import("Util_lib"); +import("Entity_lib"); -var codeCount = db.cell("select count(CONTRACTCODE) from CONTRACT where CONTRACTCODE = '" + ProcessHandlingUtil.getOnValidationValue(vars.get("$field.CONTRACTCODE")) + "'" +var contractCode, codeCount; + +contractCode = ProcessHandlingUtils.getOnValidationValue(vars.get("$field.CONTRACTCODE")); +codeCount = db.cell("select count(CONTRACTCODE) from CONTRACT where CONTRACTCODE = '" + contractCode + "'" + " and CONTRACTID <> '" + vars.get("$field.CONTRACTID") + "'"); if(codeCount > 0) { diff --git a/entity/Contract_entity/entityfields/contractdue/onValidation.js b/entity/Contract_entity/entityfields/contractdue/onValidation.js index d269674beab0906856f8e8bc83afe6ffe48c73f9..eec165e723c8199c6314723ff0e928d2f734a3ec 100644 --- a/entity/Contract_entity/entityfields/contractdue/onValidation.js +++ b/entity/Contract_entity/entityfields/contractdue/onValidation.js @@ -3,10 +3,10 @@ import("system.translate"); import("system.vars"); import("Date_lib"); import("Util_lib"); +import("Entity_lib"); -var dateUtils = new DateUtils(); -var cDue = ProcessHandlingUtil.getOnValidationValue(vars.get("$field.CONTRACTDUE")); +var cDue = ProcessHandlingUtils.getOnValidationValue(vars.get("$field.CONTRACTDUE")); -if (dateUtils.validateBeginnBeforeEnd(vars.get("$field.CONTRACTSTART"), cDue) === false || dateUtils.validateBeginnBeforeEnd(cDue, vars.get("$field.CONTRACTEND")) === false) { +if (DateUtils.validateBeginnBeforeEnd(vars.get("$field.CONTRACTSTART"), cDue) === false || DateUtils.validateBeginnBeforeEnd(cDue, vars.get("$field.CONTRACTEND")) === false) { result.string(translate.text("The next due date must be after the start of the contract and before the expiry of the contract!")); } diff --git a/entity/Contract_entity/entityfields/contractend/onValidation.js b/entity/Contract_entity/entityfields/contractend/onValidation.js index 8bb7dc15d6b77c65124e37372351a6c158d526e6..c2b570b75775ae1f669ba1a8e7ad9b716e17f2eb 100644 --- a/entity/Contract_entity/entityfields/contractend/onValidation.js +++ b/entity/Contract_entity/entityfields/contractend/onValidation.js @@ -2,10 +2,10 @@ import("system.result"); import("system.vars"); import("Date_lib"); import("Util_lib"); +import("Entity_lib"); -var dateUtils = new DateUtils(); -var cEnd = ProcessHandlingUtil.getOnValidationValue(vars.get("$field.CONTRACTEND")); +var cEnd = ProcessHandlingUtils.getOnValidationValue(vars.get("$field.CONTRACTEND")); var cDue = vars.get("$field.CONTRACTDUE"); -if (dateUtils.validateBeginnBeforeEnd(vars.get("$field.CONTRACTSTART"), cEnd) === false) - result.string(dateUtils.getValidationFailString()); \ No newline at end of file +if (DateUtils.validateBeginnBeforeEnd(vars.get("$field.CONTRACTSTART"), cEnd) === false) + result.string(DateUtils.getValidationFailString()); \ No newline at end of file diff --git a/entity/Contract_entity/entityfields/contractend/onValueChange.js b/entity/Contract_entity/entityfields/contractend/onValueChange.js index ae1ba0c5ca77f89b745d574da6d9f0c44d7013e1..b411c425e4e4a1a21d063b31fec6d81c61e9c913 100644 --- a/entity/Contract_entity/entityfields/contractend/onValueChange.js +++ b/entity/Contract_entity/entityfields/contractend/onValueChange.js @@ -1,7 +1,8 @@ import("system.vars"); import("Util_lib"); +import("Entity_lib"); -var cEnd = ProcessHandlingUtil.getOnValidationValue(vars.get("$field.CONTRACTEND")); +var cEnd = ProcessHandlingUtils.getOnValidationValue(vars.get("$field.CONTRACTEND")); var cDue = vars.get("$field.CONTRACTDUE"); if (cDue > cEnd && cDue != "") diff --git a/entity/Contract_entity/entityfields/contractstart/onValidation.js b/entity/Contract_entity/entityfields/contractstart/onValidation.js index 4ebb606eceef9c7f65fe763bb1ac5f1c62430f39..f7e82b3fd8a66a24591a7fb75a849799a3755ad4 100644 --- a/entity/Contract_entity/entityfields/contractstart/onValidation.js +++ b/entity/Contract_entity/entityfields/contractstart/onValidation.js @@ -3,10 +3,10 @@ import("system.result"); import("system.vars"); import("Date_lib"); import("Util_lib"); +import("Entity_lib"); -var dateUtils = new DateUtils(); -var cStart = ProcessHandlingUtil.getOnValidationValue(vars.get("$field.CONTRACTSTART")); +var cStart = ProcessHandlingUtils.getOnValidationValue(vars.get("$field.CONTRACTSTART")); var cDue = vars.get("$field.CONTRACTDUE"); -if (dateUtils.validateBeginnBeforeEnd(cStart, vars.get("$field.CONTRACTEND")) === false) - result.string(dateUtils.getValidationFailString()); \ No newline at end of file +if (DateUtils.validateBeginnBeforeEnd(cStart, vars.get("$field.CONTRACTEND")) === false) + result.string(DateUtils.getValidationFailString()); \ No newline at end of file diff --git a/entity/Contract_entity/entityfields/contractstart/onValueChange.js b/entity/Contract_entity/entityfields/contractstart/onValueChange.js index 2e58cfbebf5c3583bb3f18ffcc1a3d9e80b542e5..f2e29fdda8f9227616cd3dc7f3879867b2bfbaa8 100644 --- a/entity/Contract_entity/entityfields/contractstart/onValueChange.js +++ b/entity/Contract_entity/entityfields/contractstart/onValueChange.js @@ -1,7 +1,8 @@ import("system.vars"); import("Util_lib"); +import("Entity_lib"); -var cStart = ProcessHandlingUtil.getOnValidationValue(vars.get("$field.CONTRACTSTART")); +var cStart = ProcessHandlingUtils.getOnValidationValue(vars.get("$field.CONTRACTSTART")); var cDue = vars.get("$field.CONTRACTDUE"); if (cDue < cStart && cDue != "") diff --git a/entity/Offer_entity/Offer_entity.aod b/entity/Offer_entity/Offer_entity.aod index d159a36c18531a46cfa2bbf9f4ae93895ea75c3a..fb017884871b067886006739a58eb23b9aed8941 100644 --- a/entity/Offer_entity/Offer_entity.aod +++ b/entity/Offer_entity/Offer_entity.aod @@ -3,6 +3,7 @@ <name>Offer_entity</name> <title>Offer</title> <majorModelMode>DISTRIBUTED</majorModelMode> + <documentation>%aditoprj%/entity/Offer_entity/documentation.adoc</documentation> <grantUpdateProcess>%aditoprj%/entity/Offer_entity/grantUpdateProcess.js</grantUpdateProcess> <grantDeleteProcess>%aditoprj%/entity/Offer_entity/grantDeleteProcess.js</grantDeleteProcess> <recordContainerType>DB</recordContainerType> @@ -37,6 +38,7 @@ </entityField> <entityField> <name>OFFERCODE</name> + <documentation>%aditoprj%/entity/Offer_entity/entityfields/offercode/documentation.adoc</documentation> <title>Offer number</title> <tableName>OFFER</tableName> <columnName>OFFERCODE</columnName> @@ -69,6 +71,7 @@ </entityField> <entityField> <name>RELATION_ID</name> + <documentation>%aditoprj%/entity/Offer_entity/entityfields/relation_id/documentation.adoc</documentation> <title>Contact / Company</title> <tableName>OFFER</tableName> <columnName>RELATION_ID</columnName> @@ -182,6 +185,7 @@ </entityField> <entityField> <name>TotalGross</name> + <documentation>%aditoprj%/entity/Offer_entity/entityfields/totalgross/documentation.adoc</documentation> <title>Total gross</title> <state>READONLY</state> <valueProcess>%aditoprj%/entity/Offer_entity/entityfields/totalgross/valueProcess.js</valueProcess> @@ -222,6 +226,7 @@ </entityActionField> <entityField> <name>VERSNR</name> + <documentation>%aditoprj%/entity/Offer_entity/entityfields/versnr/documentation.adoc</documentation> <title>Vers. no.</title> <tableName>OFFER</tableName> <columnName>VERSNR</columnName> @@ -230,6 +235,7 @@ </entityField> <entityField> <name>OFFER_ID</name> + <documentation>%aditoprj%/entity/Offer_entity/entityfields/offer_id/documentation.adoc</documentation> </entityField> <entityFieldGroup> <name>OfferCode_VersNr</name> diff --git a/entity/Offer_entity/documentation.adoc b/entity/Offer_entity/documentation.adoc new file mode 100644 index 0000000000000000000000000000000000000000..c11de6b80022e744fb25f0e547aa69c66eeb3d00 --- /dev/null +++ b/entity/Offer_entity/documentation.adoc @@ -0,0 +1,15 @@ += OFFER + +:hardbreaks: + +Entity contains meta data of the Offer module. +It provides data from database table _OFFER_. + +== Actions + +* _Copy offer_: All values, offeritems included, are copyed and inserted in a new offer (new offer code, versnr. _1_). +* _New offer version_: All values, offeritems included, are copyed and inserted in a **new offer version** (versnr. + _1_). + +== Submodules + +* Offeritem \ No newline at end of file diff --git a/entity/Offer_entity/entityfields/newoffer/onActionProcess.js b/entity/Offer_entity/entityfields/newoffer/onActionProcess.js index 8d44f51deacd1ece4560b02cdb57803cb104cdfa..46d81aed97b3bd2e86b855020476b9a11e0618b1 100644 --- a/entity/Offer_entity/entityfields/newoffer/onActionProcess.js +++ b/entity/Offer_entity/entityfields/newoffer/onActionProcess.js @@ -2,9 +2,9 @@ import("system.logging"); import("system.vars"); import("system.neon"); import("Util_lib"); +import("Neon_lib"); import("OfferOrder_lib"); -var CMUtils = new CopyModuleUtils(); var offUtils = new OfferUtils(); var InputMapping = { @@ -24,6 +24,6 @@ var InputMapping = { } } -var ModulesMapping = CMUtils.copyModule(InputMapping); +var ModulesMapping = CopyModuleUtils.copyModule(InputMapping); -CMUtils.openNewModules("Offer_context", ModulesMapping); \ No newline at end of file +CopyModuleUtils.openNewModules("Offer_context", ModulesMapping); \ No newline at end of file diff --git a/entity/Offer_entity/entityfields/newofferversion/onActionProcess.js b/entity/Offer_entity/entityfields/newofferversion/onActionProcess.js index 60eaab09684e3967b9b0c72763695ccb5a88464c..7482f38129f497c9040811e120752578a9498a44 100644 --- a/entity/Offer_entity/entityfields/newofferversion/onActionProcess.js +++ b/entity/Offer_entity/entityfields/newofferversion/onActionProcess.js @@ -2,9 +2,9 @@ import("system.logging"); import("system.vars"); import("system.neon"); import("Util_lib"); +import("Neon_lib"); import("OfferOrder_lib"); -var CMUtils = new CopyModuleUtils(); var offUtils = new OfferUtils(); var InputMapping = { @@ -24,6 +24,6 @@ var InputMapping = { } } -var ModulesMapping = CMUtils.copyModule(InputMapping); +var ModulesMapping = CopyModuleUtils.copyModule(InputMapping); -CMUtils.openNewModules("Offer_context", ModulesMapping); \ No newline at end of file +CopyModuleUtils.openNewModules("Offer_context", ModulesMapping); \ No newline at end of file diff --git a/entity/Offer_entity/entityfields/offer_id/documentation.adoc b/entity/Offer_entity/entityfields/offer_id/documentation.adoc new file mode 100644 index 0000000000000000000000000000000000000000..45c579ffee22ce5cfa74223c23851653a2bb8750 --- /dev/null +++ b/entity/Offer_entity/entityfields/offer_id/documentation.adoc @@ -0,0 +1,4 @@ += OFFER_ID +:hardbreaks: + +This field contains link information to previous offer version. \ No newline at end of file diff --git a/entity/Offer_entity/entityfields/offercode/documentation.adoc b/entity/Offer_entity/entityfields/offercode/documentation.adoc new file mode 100644 index 0000000000000000000000000000000000000000..019d2f6339229de4bf8ba5b485bcba702cb0c6cb --- /dev/null +++ b/entity/Offer_entity/entityfields/offercode/documentation.adoc @@ -0,0 +1,5 @@ += OFFERCODE +:hardbreaks: + +Unique identification number of the offer. +It's automatically generated from the starting number _1000_. \ No newline at end of file diff --git a/entity/Offer_entity/entityfields/offercode/onValidation.js b/entity/Offer_entity/entityfields/offercode/onValidation.js index c2f733568d6821237b66fdd31a38a894370a3d18..63c353b420b443050d33dc23602eda47a08da9c8 100644 --- a/entity/Offer_entity/entityfields/offercode/onValidation.js +++ b/entity/Offer_entity/entityfields/offercode/onValidation.js @@ -3,11 +3,12 @@ import("system.result"); import("system.neon"); import("OfferOrder_lib"); import("Util_lib"); +import("Entity_lib"); var OfferUtils = new OfferUtils(); if( vars.get("$sys.operatingstate") == neon.OPERATINGSTATE_NEW - && !OfferUtils.validateOfferNumber(ProcessHandlingUtil.getOnValidationValue(vars.get("$field.OFFERCODE"))) ) + && !OfferUtils.validateOfferNumber(ProcessHandlingUtils.getOnValidationValue(vars.get("$field.OFFERCODE"))) ) { vars.set( "$field.OFFERCODE", SalesprojectUtils.getNextOfferNumber().toString() ); } \ No newline at end of file diff --git a/entity/Offer_entity/entityfields/offerdate/valueProcess.js b/entity/Offer_entity/entityfields/offerdate/valueProcess.js index af2c1f09cc605dfe6664dd0c9829026fc6785993..aeddd594f71497dedb45c4c73862b6c4dd78d440 100644 --- a/entity/Offer_entity/entityfields/offerdate/valueProcess.js +++ b/entity/Offer_entity/entityfields/offerdate/valueProcess.js @@ -5,7 +5,6 @@ import("Date_lib"); if(vars.get("$sys.operatingstate") == neon.OPERATINGSTATE_NEW && vars.get("$this.value") == "") { - var DateUtils = new DateUtils(); result.string(DateUtils.getTodayUTC()); } else diff --git a/entity/Offer_entity/entityfields/relation_id/documentation.adoc b/entity/Offer_entity/entityfields/relation_id/documentation.adoc new file mode 100644 index 0000000000000000000000000000000000000000..e978c88ef79f2554ac4731f7d4c3eb17aa50ea8b --- /dev/null +++ b/entity/Offer_entity/entityfields/relation_id/documentation.adoc @@ -0,0 +1,10 @@ += RELATION_ID +:hardbreaks: + +Over this field the company or contact person of customer is linked. + +== onValueChange + +If set in the selected company or contact person, the following field value are taken over: + +* LANGUAGE \ No newline at end of file diff --git a/entity/Offer_entity/entityfields/relation_id/onValueChange.js b/entity/Offer_entity/entityfields/relation_id/onValueChange.js index 4c3220606c52acb4eb2ee09d69427060bbd987e8..9e8a557d9112443d5bb050c2df798542b21dbbd4 100644 --- a/entity/Offer_entity/entityfields/relation_id/onValueChange.js +++ b/entity/Offer_entity/entityfields/relation_id/onValueChange.js @@ -1,8 +1,9 @@ import("system.vars"); import("system.db"); import("Util_lib"); +import("Entity_lib"); -var relid = ProcessHandlingUtil.getOnValidationValue(vars.get("$field.RELATION_ID")); +var relid = ProcessHandlingUtils.getOnValidationValue(vars.get("$field.RELATION_ID")); if(relid != "") { var relData = db.array(db.ROW, "select LANGUAGE from RELATION where RELATIONID = '" + relid + "'"); diff --git a/entity/Offer_entity/entityfields/totalgross/documentation.adoc b/entity/Offer_entity/entityfields/totalgross/documentation.adoc new file mode 100644 index 0000000000000000000000000000000000000000..c727abed66886d38bd46c6df6e1045b82666eb30 --- /dev/null +++ b/entity/Offer_entity/entityfields/totalgross/documentation.adoc @@ -0,0 +1,6 @@ += TotalGross +:hardbreaks: + +Contains the total gross of the offer (net). +This is calculated as follows: _NET_ + _VAT_. +Both are calculated in *_Offeritem_entity_* at _onDBInsert_, _onDBUpdate_ and _onDBDelete_. \ No newline at end of file diff --git a/entity/Offer_entity/entityfields/versnr/documentation.adoc b/entity/Offer_entity/entityfields/versnr/documentation.adoc new file mode 100644 index 0000000000000000000000000000000000000000..0d9cdfe8dd6e5f417025786617c8df67e43d9520 --- /dev/null +++ b/entity/Offer_entity/entityfields/versnr/documentation.adoc @@ -0,0 +1,8 @@ += VERSNR +:hardbreaks: + +Additional to the offercode there's a version number. +By default it contains the value _1_. + +A new offer version can be created over the action _newOfferVersion_. +Then the new value is the old one incremented by _1_. \ No newline at end of file diff --git a/entity/Offeritem_entity/Offeritem_entity.aod b/entity/Offeritem_entity/Offeritem_entity.aod index f9636b1e8709caf576c908bcba3680a4458bf93e..dfa1a79e76789c678181e5b94f12d8387578636b 100644 --- a/entity/Offeritem_entity/Offeritem_entity.aod +++ b/entity/Offeritem_entity/Offeritem_entity.aod @@ -3,6 +3,7 @@ <name>Offeritem_entity</name> <title>Offeritem</title> <majorModelMode>DISTRIBUTED</majorModelMode> + <documentation>%aditoprj%/entity/Offeritem_entity/documentation.adoc</documentation> <grantCreateProcess>%aditoprj%/entity/Offeritem_entity/grantCreateProcess.js</grantCreateProcess> <grantUpdateProcess>%aditoprj%/entity/Offeritem_entity/grantUpdateProcess.js</grantUpdateProcess> <grantDeleteProcess>%aditoprj%/entity/Offeritem_entity/grantDeleteProcess.js</grantDeleteProcess> @@ -117,15 +118,16 @@ </entityField> <entityField> <name>PRODUCT_ID</name> + <documentation>%aditoprj%/entity/Offeritem_entity/entityfields/product_id/documentation.adoc</documentation> <title>Article</title> <tableName>OFFERITEM</tableName> <columnName>PRODUCT_ID</columnName> <possibleItemsProcess>%aditoprj%/entity/Offeritem_entity/entityfields/product_id/possibleItemsProcess.js</possibleItemsProcess> - <onValidation>%aditoprj%/entity/Offeritem_entity/entityfields/product_id/onValidation.js</onValidation> <onValueChange>%aditoprj%/entity/Offeritem_entity/entityfields/product_id/onValueChange.js</onValueChange> </entityField> <entityField> <name>QUANTITY</name> + <documentation>%aditoprj%/entity/Offeritem_entity/entityfields/quantity/documentation.adoc</documentation> <title>Quantity</title> <tableName>OFFERITEM</tableName> <columnName>QUANTITY</columnName> @@ -180,6 +182,7 @@ </entityParameter> <entityField> <name>TotalPrice</name> + <documentation>%aditoprj%/entity/Offeritem_entity/entityfields/totalprice/documentation.adoc</documentation> <title>Sum</title> <contentType>NUMBER</contentType> <outputFormat>#,##0.00</outputFormat> diff --git a/entity/Offeritem_entity/documentation.adoc b/entity/Offeritem_entity/documentation.adoc new file mode 100644 index 0000000000000000000000000000000000000000..a5283456a0759e860a058690aeb402321c2779ec --- /dev/null +++ b/entity/Offeritem_entity/documentation.adoc @@ -0,0 +1,21 @@ += OFFERITEM +:hardbreaks: + +Entity contains products that belong to the linked offer. +It provides data from database table _OFFERITEM_ by a JDito record container. + +== onDBInsert + +If a new offer item is to be added, the parts list of the corresponding product +is inserted at the same time. A new item is created for each product. +The net sum in the linked offer is updated. + +== onDBUpdate + +The net sum in the linked offer is updated. + + +== onDBDelete + +Before deletion of an item, its subordinated items are also deleted. +The net sum in the linked offer is updated. \ No newline at end of file diff --git a/entity/Offeritem_entity/entityfields/product_id/documentation.adoc b/entity/Offeritem_entity/entityfields/product_id/documentation.adoc new file mode 100644 index 0000000000000000000000000000000000000000..5d6ffbd8dd98f0a218cec7b0d8a0228b1e65b94e --- /dev/null +++ b/entity/Offeritem_entity/entityfields/product_id/documentation.adoc @@ -0,0 +1,14 @@ += PRODUCT_ID +:hardbreaks: + +Contains link to selected product for this item. + +== onValueChange + +The following field are set with the values from the selected product: + +* GROUPCODEID +* UNIT +* ITEMNAME +* PRICE & VAT: Values from the current valid price list defined to this product. + For futher informations see the comments in method __getPriceListToUse_ in _Product_lib_. \ No newline at end of file diff --git a/entity/Offeritem_entity/entityfields/product_id/onValidation.js b/entity/Offeritem_entity/entityfields/product_id/onValidation.js deleted file mode 100644 index e69de29bb2d1d6434b8b29ae775ad8c2e48c5391..0000000000000000000000000000000000000000 diff --git a/entity/Offeritem_entity/entityfields/product_id/onValueChange.js b/entity/Offeritem_entity/entityfields/product_id/onValueChange.js index 196b4ab169ea3cbfdb3f6e1feb6d341ce6a008ae..252419a862f0599a975424643fd12bd2a60f1958 100644 --- a/entity/Offeritem_entity/entityfields/product_id/onValueChange.js +++ b/entity/Offeritem_entity/entityfields/product_id/onValueChange.js @@ -3,8 +3,9 @@ import("system.vars"); import("system.neon"); import("Product_lib"); import("Util_lib"); +import("Entity_lib"); -var pid = ProcessHandlingUtil.getOnValidationValue(vars.get("$field.PRODUCT_ID")); +var pid = ProcessHandlingUtils.getOnValidationValue(vars.get("$field.PRODUCT_ID")); if(pid != "") { var curr = vars.exists("$param.Currency_param") ? vars.get("$param.Currency_param") : ""; diff --git a/entity/Offeritem_entity/entityfields/quantity/documentation.adoc b/entity/Offeritem_entity/entityfields/quantity/documentation.adoc new file mode 100644 index 0000000000000000000000000000000000000000..5f7419c93c2913f83e1e1ab76f14dd1f03e49d2c --- /dev/null +++ b/entity/Offeritem_entity/entityfields/quantity/documentation.adoc @@ -0,0 +1,9 @@ += QUANTITY +:hardbreaks: + +Quantity of selected product in this item. + +== onValueChange + +The fields PRICE & VAT are set by changing quantity when a product has already been selected. +The reason for this is that you can define a quantity as of which price lists are valid. \ No newline at end of file diff --git a/entity/Offeritem_entity/entityfields/quantity/onValueChange.js b/entity/Offeritem_entity/entityfields/quantity/onValueChange.js index d605b64c6f8ad78ba76189a0333d426011f7e337..aedc968cc70beb0c78061c5d2848f9fcc89caed3 100644 --- a/entity/Offeritem_entity/entityfields/quantity/onValueChange.js +++ b/entity/Offeritem_entity/entityfields/quantity/onValueChange.js @@ -3,9 +3,10 @@ import("system.vars"); import("system.neon"); import("Product_lib"); import("Util_lib"); +import("Entity_lib"); var pid = vars.get("$field.PRODUCT_ID"); -var newQuantity = ProcessHandlingUtil.getOnValidationValue(vars.get("$field.QUANTITY")); +var newQuantity = ProcessHandlingUtils.getOnValidationValue(vars.get("$field.QUANTITY")); if(pid != "" && newQuantity != "") { var curr = vars.exists("$param.Currency_param") ? vars.get("$param.Currency_param") : ""; diff --git a/entity/Offeritem_entity/entityfields/totalprice/documentation.adoc b/entity/Offeritem_entity/entityfields/totalprice/documentation.adoc new file mode 100644 index 0000000000000000000000000000000000000000..481fb639fb88ae3d95aa88fa8cf0cee9160dc2e5 --- /dev/null +++ b/entity/Offeritem_entity/entityfields/totalprice/documentation.adoc @@ -0,0 +1,5 @@ += TotalPrice +:hardbreaks: + +Contains the total price of the item. +This is calculated as follows: _QUANTITY_ * _PRICE_ * (100 - _DISCOUNT_) / 100 \ No newline at end of file diff --git a/entity/Prod2prod_entity/Prod2prod_entity.aod b/entity/Prod2prod_entity/Prod2prod_entity.aod index 5966912c271c770f8c6f9f04cb4b9bfc3f8e9a5b..ce05f40bd6b8211c3c09efc9b048a706a6fbf6c1 100644 --- a/entity/Prod2prod_entity/Prod2prod_entity.aod +++ b/entity/Prod2prod_entity/Prod2prod_entity.aod @@ -2,6 +2,7 @@ <entity xmlns="http://www.adito.de/2018/ao/Model" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" VERSION="1.0.5" xsi:schemaLocation="http://www.adito.de/2018/ao/Model adito://models/xsd/entity/1.0.5"> <name>Prod2prod_entity</name> <majorModelMode>DISTRIBUTED</majorModelMode> + <documentation>%aditoprj%/entity/Prod2prod_entity/documentation.adoc</documentation> <externalOpenAction>%aditoprj%/entity/Prod2prod_entity/externalOpenAction.js</externalOpenAction> <recordContainerType>JDITO</recordContainerType> <alias>Data_alias</alias> @@ -49,6 +50,7 @@ </entityField> <entityField> <name>DEST_ID</name> + <documentation>%aditoprj%/entity/Prod2prod_entity/entityfields/dest_id/documentation.adoc</documentation> <tableName></tableName> <columnName></columnName> <fieldName>DEST_ID</fieldName> @@ -81,6 +83,7 @@ </entityField> <entityField> <name>SOURCE_ID</name> + <documentation>%aditoprj%/entity/Prod2prod_entity/entityfields/source_id/documentation.adoc</documentation> <title>Product</title> <tableName></tableName> <columnName></columnName> @@ -90,6 +93,7 @@ </entityField> <entityField> <name>TAKEPRICE</name> + <documentation>%aditoprj%/entity/Prod2prod_entity/entityfields/takeprice/documentation.adoc</documentation> <title>Price</title> <tableName></tableName> <columnName></columnName> diff --git a/entity/Prod2prod_entity/documentation.adoc b/entity/Prod2prod_entity/documentation.adoc new file mode 100644 index 0000000000000000000000000000000000000000..59bbbae97ad3a477420b71e111fac26c5dada832 --- /dev/null +++ b/entity/Prod2prod_entity/documentation.adoc @@ -0,0 +1,34 @@ += Prod2prod_entity +:hardbreaks: + +Entity contains links between products. +It provides data from database table _PROD2PROD_. + +The record container loads the parts list of the passed product. +This means which sub-products this product consists of. + +A possible **definiton** for a parts list is: + +[quote, McGraw-Hill Dictionary of Scientific & Technical Terms, 6E. S.v. "parts list." Retrieved November 26 2018 from https://encyclopedia2.thefreedictionary.com/parts+list] +____ +One or more printed sheets showing a manufacturer's parts or assemblies of an end item by illustration or a numerical listing of part numbers and names; it does not outline any assembly, maintenance, or operating instructions, and it may or may not have a price list cover sheet. +____ + +Database table _PROD2PROD_ contains the two foreign key columns _DEST_ID_ and _SOURCE_ID_. +Both contain a _PRODUCTID_. The following visualization clarifies the structure of a parts list. + +++++ +<img src="" style="cursor:pointer;max-width:100%;" onclick="(function(img){if(img.wnd!=null&&!img.wnd.closed){img.wnd.focus();}else{var r=function(evt){if(evt.data=='ready'&&evt.source==img.wnd){img.wnd.postMessage(decodeURIComponent(img.getAttribute('src')),'*');window.removeEventListener('message',r);}};window.addEventListener('message',r);img.wnd=window.open('https://www.draw.io/?client=1&lightbox=1&edit=_blank');}})(this);"/> +++++ + +//// +How to edit this diagram: +1. copy complete <img> tag from above +2. paste it into text editor and save it as a html file +3. open html file with browser +4. click on the diagram +5. at the bottom of the page you can start edit mode +6. edit diagram on draw.io +7. get <img> tag of the diagram with File -> Embed -> Image +8. replace tag above +//// \ No newline at end of file diff --git a/entity/Prod2prod_entity/entityfields/dest_id/documentation.adoc b/entity/Prod2prod_entity/entityfields/dest_id/documentation.adoc new file mode 100644 index 0000000000000000000000000000000000000000..34ed7bcc2d426001e59818f56d45016fdb521c6e --- /dev/null +++ b/entity/Prod2prod_entity/entityfields/dest_id/documentation.adoc @@ -0,0 +1,7 @@ += DEST_ID +:hardbreaks: + +This column contains link information (foreign key) to the product to which it is a part of (parent). + +== Preset New +The system defaults to the passed value of the parameter _ProductId_param_. \ No newline at end of file diff --git a/entity/Prod2prod_entity/entityfields/optional/documentation.adoc b/entity/Prod2prod_entity/entityfields/optional/documentation.adoc new file mode 100644 index 0000000000000000000000000000000000000000..3ca64e8047d84398535b3cfa2d20919c5ecd4e25 --- /dev/null +++ b/entity/Prod2prod_entity/entityfields/optional/documentation.adoc @@ -0,0 +1,4 @@ += OPTIONAL +:hardbreaks: + +Flag for offer / order. The value is taken over in offer / order. \ No newline at end of file diff --git a/entity/Prod2prod_entity/entityfields/source_id/documentation.adoc b/entity/Prod2prod_entity/entityfields/source_id/documentation.adoc new file mode 100644 index 0000000000000000000000000000000000000000..59dcd2e0f4c01785c34f531b3f43bd54235841f6 --- /dev/null +++ b/entity/Prod2prod_entity/entityfields/source_id/documentation.adoc @@ -0,0 +1,8 @@ += SOURCE_ID +:hardbreaks: + +This column contains link information (foreign key) to the partial product. (child) + +== value list +Listed are all products which are **not already part of** parts list. +This means each product recursively appears in _DEST_ID_ or _SOURCE_ID_ of the current product passed by parameter _ProductId_param_ **is excluded**. \ No newline at end of file diff --git a/entity/Prod2prod_entity/entityfields/takeprice/documentation.adoc b/entity/Prod2prod_entity/entityfields/takeprice/documentation.adoc new file mode 100644 index 0000000000000000000000000000000000000000..eb617876677f893ff5d953b92f7cd5cff145f60e --- /dev/null +++ b/entity/Prod2prod_entity/entityfields/takeprice/documentation.adoc @@ -0,0 +1,4 @@ += TAKEPRICE +:hardbreaks: + +Flag for offer / order. If set the current valid price list to this product is taken over in offer / order. \ No newline at end of file diff --git a/entity/Product_entity/Product_entity.aod b/entity/Product_entity/Product_entity.aod index 439bbabafd0959b155035cc79d979774c422fcfe..c0010e3e41a6303993fb2d67197ca7e828b0ff60 100644 --- a/entity/Product_entity/Product_entity.aod +++ b/entity/Product_entity/Product_entity.aod @@ -3,6 +3,7 @@ <name>Product_entity</name> <title>Product</title> <majorModelMode>DISTRIBUTED</majorModelMode> + <documentation>%aditoprj%/entity/Product_entity/documentation.adoc</documentation> <recordContainerType>DB</recordContainerType> <iconId>VAADIN:HAMMER</iconId> <alias>Data_alias</alias> @@ -60,6 +61,7 @@ </entityField> <entityField> <name>PRODUCTCODE</name> + <documentation>%aditoprj%/entity/Product_entity/entityfields/productcode/documentation.adoc</documentation> <title>Product number</title> <tableName>PRODUCT</tableName> <columnName>PRODUCTCODE</columnName> diff --git a/entity/Product_entity/documentation.adoc b/entity/Product_entity/documentation.adoc new file mode 100644 index 0000000000000000000000000000000000000000..ccd3865898d44c3d0ebd4b3bd46633d4f84b8bf1 --- /dev/null +++ b/entity/Product_entity/documentation.adoc @@ -0,0 +1,12 @@ += PRODUCT + +:hardbreaks: + +Entity contains meta data of the Product module. +It provides data from database table _PRODUCT_. + +== Submodules + +* Parts list (Prod2Prod) +* Product price +* Stock \ No newline at end of file diff --git a/entity/Product_entity/entityfields/productcode/documentation.adoc b/entity/Product_entity/entityfields/productcode/documentation.adoc new file mode 100644 index 0000000000000000000000000000000000000000..e3dd7742201c099ba3c7af84b92d80b5c9444479 --- /dev/null +++ b/entity/Product_entity/entityfields/productcode/documentation.adoc @@ -0,0 +1,4 @@ += PRODUCTCODE +:hardbreaks: + +Unique identification number of the product. (max. 30 characters, not automatically generated) \ No newline at end of file diff --git a/entity/Product_entity/entityfields/productcode/onValidation.js b/entity/Product_entity/entityfields/productcode/onValidation.js index b72108499db871aa2345dd05649f4cf0178edbc5..1dc5b4f996393023dd1f9e0f4070aad491cf515b 100644 --- a/entity/Product_entity/entityfields/productcode/onValidation.js +++ b/entity/Product_entity/entityfields/productcode/onValidation.js @@ -3,8 +3,12 @@ import("system.result"); import("system.vars"); import("system.db"); import("Util_lib"); +import("Entity_lib"); -var codeCount = db.cell("select count(PRODUCTCODE) from PRODUCT where PRODUCTCODE = '" + ProcessHandlingUtil.getOnValidationValue(vars.get("$field.PRODUCTCODE")) + "'" +var codeCount, productCode; + +productCode = ProcessHandlingUtils.getOnValidationValue(vars.get("$field.PRODUCTCODE")); +codeCount = db.cell("select count(PRODUCTCODE) from PRODUCT where PRODUCTCODE = '" + productCode + "'" + " and PRODUCTID <> '" + vars.get("$field.PRODUCTID") + "'"); if(codeCount > 0) { diff --git a/entity/Productprice_entity/Productprice_entity.aod b/entity/Productprice_entity/Productprice_entity.aod index 0897e500a76be11b22381413ef3a4c312199ea08..039922c708f81775e5285feb0d1fd9cc8af6af88 100644 --- a/entity/Productprice_entity/Productprice_entity.aod +++ b/entity/Productprice_entity/Productprice_entity.aod @@ -135,9 +135,11 @@ </entityField> <entityField> <name>PRICELIST</name> + <documentation>%aditoprj%/entity/Productprice_entity/entityfields/pricelist/documentation.adoc</documentation> <title>Price list</title> <tableName>PRODUCTPRICE</tableName> <columnName>PRICELIST</columnName> + <mandatoryProcess>%aditoprj%/entity/Productprice_entity/entityfields/pricelist/mandatoryProcess.js</mandatoryProcess> <possibleItemsProcess>%aditoprj%/entity/Productprice_entity/entityfields/pricelist/possibleItemsProcess.js</possibleItemsProcess> <state>AUTO</state> <stateProcess>%aditoprj%/entity/Productprice_entity/entityfields/pricelist/stateProcess.js</stateProcess> diff --git a/entity/Productprice_entity/documentation.adoc b/entity/Productprice_entity/documentation.adoc index e69de29bb2d1d6434b8b29ae775ad8c2e48c5391..cb6558ccb5adeee75778e6e5e83b152b94f590c7 100644 --- a/entity/Productprice_entity/documentation.adoc +++ b/entity/Productprice_entity/documentation.adoc @@ -0,0 +1,15 @@ += PRODUCTPRICE + +:hardbreaks: + +Entity contains price lists for the linked product. +It provides data from database table _PRODUCTPRICE_. + +== New price list via product + +By creation via product you have to select a price list type using the field _PRICELIST_. + +== New price list via organisation (custom price list) + +By creation via an organisation you don't have do select a price list type because +it's a custom price list \-> field _RELATION_ID_ is filled automatically by parameter _Relation_Id_param_. \ No newline at end of file diff --git a/entity/Productprice_entity/entityfields/pricelist/documentation.adoc b/entity/Productprice_entity/entityfields/pricelist/documentation.adoc new file mode 100644 index 0000000000000000000000000000000000000000..777173dda73190f975c75f5475f8f7a31b4d7bfb --- /dev/null +++ b/entity/Productprice_entity/entityfields/pricelist/documentation.adoc @@ -0,0 +1,18 @@ += PRICELIST + +:hardbreaks: + +Field indicates the type of the price list. + +== onValidation + +There's a **verficiation** for current pricelist whether an **identical price list already exists** for product. +The following criteria will be checked in function _ProductUtils.checkForIndenticalPriceLists()_: + +* Identical price list type +* Identical from quantity +* Identical currency +* Identical purchase price/sales price +* Idcentical Valid from and valid to OR + +If all criteria are fulfilled, price list can not be saved. \ No newline at end of file diff --git a/entity/Productprice_entity/entityfields/pricelist/mandatoryProcess.js b/entity/Productprice_entity/entityfields/pricelist/mandatoryProcess.js new file mode 100644 index 0000000000000000000000000000000000000000..6db2c72b17a7335446e477540a3117ae5704fa35 --- /dev/null +++ b/entity/Productprice_entity/entityfields/pricelist/mandatoryProcess.js @@ -0,0 +1,7 @@ +import("system.vars"); +import("system.result"); + +if(vars.get("$field.RELATION_ID") != "") + result.string("false"); +else + result.string("true"); \ No newline at end of file diff --git a/entity/Productprice_entity/entityfields/pricelist/onValidation.js b/entity/Productprice_entity/entityfields/pricelist/onValidation.js index 4927c81bd892cc93dbb387a3f8a1407982471e41..280ee4e3e23eed2cd040d924b6db4a9d482935a0 100644 --- a/entity/Productprice_entity/entityfields/pricelist/onValidation.js +++ b/entity/Productprice_entity/entityfields/pricelist/onValidation.js @@ -3,10 +3,11 @@ import("system.result"); import("system.vars"); import("Product_lib"); import("Util_lib"); +import("Entity_lib"); var pUtils = new ProductUtils(); var priceList = { - priceList: ProcessHandlingUtil.getOnValidationValue(vars.get("$field.PRICELIST")) + priceList: ProcessHandlingUtils.getOnValidationValue(vars.get("$field.PRICELIST")) , fromQuantity: vars.get("$field.FROMQUANTITY") , buySell: vars.get("$field.BUYSELL") , currency: vars.get("$field.CURRENCY") diff --git a/entity/Productprice_entity/entityfields/valid_from/onValidation.js b/entity/Productprice_entity/entityfields/valid_from/onValidation.js index 74fa861f9bb6db8bcef42cc1ee335bd281fadac6..5ebbf1a7dccee4b0a58c46b478596bf326697184 100644 --- a/entity/Productprice_entity/entityfields/valid_from/onValidation.js +++ b/entity/Productprice_entity/entityfields/valid_from/onValidation.js @@ -2,9 +2,9 @@ import("system.result"); import("system.vars"); import("Date_lib"); import("Util_lib"); +import("Entity_lib"); -var dateUtils = new DateUtils(); -var cStart = ProcessHandlingUtil.getOnValidationValue(vars.get("$field.VALID_FROM")); +var cStart = ProcessHandlingUtils.getOnValidationValue(vars.get("$field.VALID_FROM")); -if (dateUtils.validateBeginnBeforeEnd(cStart, vars.get("$field.VALID_TO")) === false) - result.string(dateUtils.getValidationFailString()); \ No newline at end of file +if (DateUtils.validateBeginnBeforeEnd(cStart, vars.get("$field.VALID_TO")) === false) + result.string(DateUtils.getValidationFailString()); \ No newline at end of file diff --git a/entity/Productprice_entity/entityfields/valid_to/onValidation.js b/entity/Productprice_entity/entityfields/valid_to/onValidation.js index de9f15393fcf97ff4efb4edd960f4aea2a268d72..3ada2dd1e76e6dceb2b4c1dbf169bf91e0125579 100644 --- a/entity/Productprice_entity/entityfields/valid_to/onValidation.js +++ b/entity/Productprice_entity/entityfields/valid_to/onValidation.js @@ -2,9 +2,9 @@ import("system.result"); import("system.vars"); import("Date_lib"); import("Util_lib"); +import("Entity_lib"); -var dateUtils = new DateUtils(); -var cEnd = ProcessHandlingUtil.getOnValidationValue(vars.get("$field.VALID_TO")); +var cEnd = ProcessHandlingUtils.getOnValidationValue(vars.get("$field.VALID_TO")); -if (dateUtils.validateBeginnBeforeEnd(vars.get("$field.VALID_FROM"), cEnd) === false) - result.string(dateUtils.getValidationFailString()); \ No newline at end of file +if (DateUtils.validateBeginnBeforeEnd(vars.get("$field.VALID_FROM"), cEnd) === false) + result.string(DateUtils.getValidationFailString()); \ No newline at end of file diff --git a/entity/SalesprojectCompetition_entity/entityfields/reason/onValidation.js b/entity/SalesprojectCompetition_entity/entityfields/reason/onValidation.js index 6a13d6a1ca6838c022617f2face7ebe539efab78..dde86e982a4f580311473d37bdb9ee9c61c8dd77 100644 --- a/entity/SalesprojectCompetition_entity/entityfields/reason/onValidation.js +++ b/entity/SalesprojectCompetition_entity/entityfields/reason/onValidation.js @@ -2,8 +2,9 @@ import("system.result"); import("system.vars"); import("system.translate"); import("Util_lib"); +import("Entity_lib"); -var reason = ProcessHandlingUtil.getOnValidationValue(vars.get("$field.REASON")); +var reason = ProcessHandlingUtils.getOnValidationValue(vars.get("$field.REASON")); if (!vars.getString("$field.DATE_CANCELLED") && reason) { result.string(translate.text("A reason is only possible if a date is given.")) diff --git a/entity/Salesproject_entity/entityfields/enddate/valueProcess.js b/entity/Salesproject_entity/entityfields/enddate/valueProcess.js index b27871853b658ad34b7de4e3e674d0a627c1617b..857933a9839949add7347dcccb862b6b40601dec 100644 --- a/entity/Salesproject_entity/entityfields/enddate/valueProcess.js +++ b/entity/Salesproject_entity/entityfields/enddate/valueProcess.js @@ -5,7 +5,6 @@ import("Date_lib"); if(vars.get("$sys.operatingstate") == neon.OPERATINGSTATE_NEW && vars.get("$this.value") == "") { - var DateUtils = new DateUtils(); result.string(DateUtils.getDateIncrementedByYears(DateUtils.getTodayUTC(), 1)); } else diff --git a/entity/Salesproject_entity/entityfields/projectcode/onValidation.js b/entity/Salesproject_entity/entityfields/projectcode/onValidation.js index d98945cc17aabeb18d7213d3ebb8e6c408d0e6ac..1f7657c4feda5d0e61dd89fe7dccffa1043efb85 100644 --- a/entity/Salesproject_entity/entityfields/projectcode/onValidation.js +++ b/entity/Salesproject_entity/entityfields/projectcode/onValidation.js @@ -3,9 +3,10 @@ import("system.result"); import("system.neon"); import("Salesproject_lib"); import("Util_lib"); +import("Entity_lib"); if( vars.get("$sys.operatingstate") == neon.OPERATINGSTATE_NEW - && !Salesproject.validateProjectNumber(ProcessHandlingUtil.getOnValidationValue(vars.get("$field.PROJECTCODE"))) ) + && !Salesproject.validateProjectNumber(ProcessHandlingUtils.getOnValidationValue(vars.get("$field.PROJECTCODE"))) ) { vars.set( "$field.PROJECTCODE", Salesproject.getNextProjectNumber().toString("1 = 1") ); } \ No newline at end of file diff --git a/entity/Salesproject_entity/entityfields/startdate/valueProcess.js b/entity/Salesproject_entity/entityfields/startdate/valueProcess.js index af2c1f09cc605dfe6664dd0c9829026fc6785993..aeddd594f71497dedb45c4c73862b6c4dd78d440 100644 --- a/entity/Salesproject_entity/entityfields/startdate/valueProcess.js +++ b/entity/Salesproject_entity/entityfields/startdate/valueProcess.js @@ -5,7 +5,6 @@ import("Date_lib"); if(vars.get("$sys.operatingstate") == neon.OPERATINGSTATE_NEW && vars.get("$this.value") == "") { - var DateUtils = new DateUtils(); result.string(DateUtils.getTodayUTC()); } else diff --git a/entity/Stock_entity/Stock_entity.aod b/entity/Stock_entity/Stock_entity.aod index 918140f62110b5d809333a1b3a4c7f677f4ee9d3..dfae4c28d432d6336ce785d7b4d71155351ba1a1 100644 --- a/entity/Stock_entity/Stock_entity.aod +++ b/entity/Stock_entity/Stock_entity.aod @@ -3,6 +3,7 @@ <name>Stock_entity</name> <title>Stock</title> <majorModelMode>DISTRIBUTED</majorModelMode> + <documentation>%aditoprj%/entity/Stock_entity/documentation.adoc</documentation> <recordContainerType>DB</recordContainerType> <alias>Data_alias</alias> <conditionProcess>%aditoprj%/entity/Stock_entity/conditionProcess.js</conditionProcess> diff --git a/entity/Stock_entity/documentation.adoc b/entity/Stock_entity/documentation.adoc new file mode 100644 index 0000000000000000000000000000000000000000..988a6942455abacabdd88bb4170f27b36ed446e5 --- /dev/null +++ b/entity/Stock_entity/documentation.adoc @@ -0,0 +1,6 @@ += STOCK + +:hardbreaks: + +Entity contains stock for the linked product. +It provides data from database table _STOCK_. \ No newline at end of file diff --git a/entity/Stock_entity/entityfields/stockcount/valueProcess.js b/entity/Stock_entity/entityfields/stockcount/valueProcess.js index d5de29c8e34a1b58595736e30155324f168ff5b5..4c32e2524eda69478457c1979858e40229135cba 100644 --- a/entity/Stock_entity/entityfields/stockcount/valueProcess.js +++ b/entity/Stock_entity/entityfields/stockcount/valueProcess.js @@ -5,7 +5,7 @@ import("Product_lib"); if(vars.exists("$param.ProductId_param") && vars.get("$param.ProductId_param") != "") { - var ProductUtils = new ProductUtils(); + var pUtils = new ProductUtils(); - result.string( ProductUtils.getStockCount(vars.get("$param.ProductId_param")) ); + result.string( pUtils.getStockCount(vars.get("$param.ProductId_param")) ); } \ No newline at end of file diff --git a/others/guide/how to write JDito code.adoc b/others/guide/how to write JDito code.adoc index 7e71a092f62e6fdd0ee34c4c77d5e6be4ee5c6eb..79d6a489bccec838c897e390e6607cdb895c69b3 100644 --- a/others/guide/how to write JDito code.adoc +++ b/others/guide/how to write JDito code.adoc @@ -1,5 +1,5 @@ How to write JDito code -============================ +======================= :toc2: left :numbered: @@ -26,11 +26,27 @@ data = vars.get....... Example: [source,javascript] ---- -for (i = 0, i < dataLen; i++){ +for (i = 0, i < dataLen; i++) { //code here } ---- +=== loops === +nested loops should be defined with replicated indexer variables (or with a good and describing name) + +Therefore it's easy to see in which level of the counter you are. + +Example: +[source,javascript] +---- +for (i = 0, i < dataLen; i++) { + for (ii = 0, ii < dataLen[i].length; ii++) { + //code... + } +} +---- + + == defining different types of functions n libraries == === by using static methods === diff --git a/others/guide/how to write a library.adoc b/others/guide/how to write a library.adoc new file mode 100644 index 0000000000000000000000000000000000000000..a3175cc3f706693f1bcf5664c997e095dd5362f4 --- /dev/null +++ b/others/guide/how to write a library.adoc @@ -0,0 +1,110 @@ +How to write a JDito library +============================ +:toc2: left +:numbered: + + +== defining different types of functions == + +=== by using static methods === +This will be mostly utility functions and so on, where there is no need to instanciate an object. You'll need this probably the most time. + + +Definition: +[source,javascript] +---- +/** + * provides static methods for validation of communication data + * do not create an instance of this + * @static + * @class + */ +function CommValidationUtil(){<1> +} + +/** + * returns a blueprint for validation extensions; these extensions are needed for validating comm data and can be passed to other functions + * @return {object} a object with properties that have a specific default value; normally you want to overwrite that value + */ +CommValidationUtil.getExtensionsBlueprint = function(){<2> + return { + countryCode: null + }; +} +---- +<1> the function-object that keeps everything together - this function should never be actually called (no direct call, no indirect call) +<2> an actual function that can be called + +And how to use it: +[source,javascript] +---- +import("Comm_lib"); + +var additionals = CommValidationUtil.getExtensionsBlueprint(); +---- + +=== by creating an object with functions === + +You may want to hold data and create objects where methods share that data. + +Definition: +[source,javascript] +---- +/** + * object for easier handling of conditions; <1> + * With this object you do not have to check if the string is empty or not; + * you don't need to append a "1=1" condition or similar; + * this objects gains most benefit if you have a lot of conditions that are added (or not) depending on tons of JDito-conditions + * @class + * @param {String} [alias=the current alias] the database alias where the condition shall be executed later (important for column types of preparedStatements) <2> + * @example //TODO: add missing example <3> + */ +function SqlCondition(alias){<4> + //setting null is only needed to provide autocomplete for the ADITO-designer + this.preparedValues = null; + this._init();//the properties are initalized in an extra function because init is nearly the same as resetting (clearing) the SqlConditions + this.alias = alias; +} +/** + * append with SQL-and; no paranthesize of existing conditions is done + * @param {String} cond the condition string which shall be appended + * @return {Object} current SqlCondition-object + */ +SqlCondition.prototype.and = function(cond){<5> + if (!cond) + return this; + if (this._sqlStorage) + this._sqlStorage += " and "; + this._sqlStorage += cond; + return this; +} +---- + +<1> JS-Doc comment: http://usejsdoc.org/ +<2> use the correct form for optional/required parameters: http://usejsdoc.org/tags-param.html +<3> examples are usefull on more complex functions +<4> constructor function; init propiertes (do not set functions ("methods") here!) +<5> add functions ("methods") to the prototype, they are available through the prototype chain + +And how to use it (normally you'd want to use preparedStatements but for the sake of an easy example it's a bit shorter here) +[source,javascript] +---- +import("system.vars"); +import("system.result"); +import("Sql_lib"); +import("Comm_lib"); + +var cond, mediumIds, idVal; + +cond = new SqlCondition(); + +mediumIds = CommExtensions.getContextualMediumIds(); +if (mediumIds.length > 0) + cond.and("COMM.MEDIUM_ID in (" + mediumIds.join(", ") + ")"); + +idVal = vars.get("$local.idvalue"); +if (uids.length > 0) + cond.and("COMM.COMMID = '" + idVal + "' "); + +result.string(cond.toString("COMM.OPTIONAL = 't'")); +---- \ No newline at end of file diff --git a/process/Date_lib/process.js b/process/Date_lib/process.js index 18a8c5d75a362aa1c4c552ee1cb394267be39062..b91e9b129d54a95f3e9c00c637daf7797da40f48 100644 --- a/process/Date_lib/process.js +++ b/process/Date_lib/process.js @@ -5,45 +5,44 @@ import("system.datetime"); * provides methods for interactions with dates */ function DateUtils(){ - - var that = this; - /** - * Validates two date inputs (beginning should always be before the end!) - * - * @param pStart {Number} - * @param pEnd {Number} - * - * @result {Boolean|null} Boolean if it was able to check smth or null if the input values were not valid - */ - this.validateBeginnBeforeEnd = function(pStart, pEnd) { - if (pStart == "" || pStart == null || pEnd == "" || pEnd == null) return null; - return pStart <= pEnd; - } - - this.getValidationFailString = function(){ - return translate.text("The expiry date must be after the start date!"); - } - - /** - * Delivers the current date at midnight in UTC - * - * @result {Number} - */ - this.getTodayUTC = function(){ - return datetime.today("UTC"); - } - - /** - * Delivers the passed date incremented by passed years - * - * @param pDate {Number} - * @param pYears {Number} - * - * @result {Number} date incremented by years - */ - this.getDateIncrementedByYears = function(pDate, pYears){ - var dateObj = new Date(pDate); - - return dateObj.setFullYear(dateObj.getFullYear() + pYears); - } +} + +/** + * Validates two date inputs (beginning should always be before the end!) + * + * @param pStart {Number} + * @param pEnd {Number} + * + * @result {Boolean|null} Boolean if it was able to check smth or null if the input values were not valid + */ +DateUtils.validateBeginnBeforeEnd = function(pStart, pEnd) { + if (pStart == "" || pStart == null || pEnd == "" || pEnd == null) return null; + return pStart <= pEnd; +} + +DateUtils.getValidationFailString = function(){ + return translate.text("The expiry date must be after the start date!"); +} + +/** + * Delivers the current date at midnight in UTC + * + * @result {Number} + */ +DateUtils.getTodayUTC = function(){ + return datetime.today("UTC"); +} + +/** + * Delivers the passed date incremented by passed years + * + * @param pDate {Number} + * @param pYears {Number} + * + * @result {Number} date incremented by years + */ +DateUtils.getDateIncrementedByYears = function(pDate, pYears){ + var dateObj = new Date(pDate); + + return dateObj.setFullYear(dateObj.getFullYear() + pYears); } \ No newline at end of file diff --git a/process/Entity_lib/Entity_lib.aod b/process/Entity_lib/Entity_lib.aod new file mode 100644 index 0000000000000000000000000000000000000000..d8eaf5d930b5b52d7ffac81d99591fbecf8dc56c --- /dev/null +++ b/process/Entity_lib/Entity_lib.aod @@ -0,0 +1,6 @@ +<?xml version="1.0" encoding="UTF-8"?> +<process xmlns="http://www.adito.de/2018/ao/Model" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" VERSION="1.1.7" xsi:schemaLocation="http://www.adito.de/2018/ao/Model adito://models/xsd/process/1.1.7"> + <name>Entity_lib</name> + <majorModelMode>DISTRIBUTED</majorModelMode> + <process>%aditoprj%/process/Entity_lib/process.js</process> +</process> diff --git a/process/Entity_lib/process.js b/process/Entity_lib/process.js new file mode 100644 index 0000000000000000000000000000000000000000..4c0e41ed076844b898fe86edc950773ceb3d1d15 --- /dev/null +++ b/process/Entity_lib/process.js @@ -0,0 +1,23 @@ +/** +* provides static methods for special handling of entities in JDito-Processes +* do not create an instance of this +* @class +* @static +*/ +function ProcessHandlingUtils(){ +} + +/** +* In onValidation-Process a local variable called "$local.value" is made available from kernel. +* It contains the entered value - the field contains the value only after successfull validation (vars.get("$field.Fieldname")). +* The onValidation-Process is running again before saving the entity, at this point there's "$local.value" varialbe no longer available, +* but the entered value now is the present one because the field has already been validated before. +* Otherwise a "variable not found" error would occur. +* +* @param {String} fieldValue value of the field onValidation-Process is executed ( e.g. vars.get("$field.Fieldname") ) +* +* @return {String} Field value for onValidation-Process +*/ +ProcessHandlingUtils.getOnValidationValue = function(fieldValue) { + return vars.exists("$local.value") ? vars.get("$local.value") : fieldValue; +} diff --git a/process/IndexSearch_lib/process.js b/process/IndexSearch_lib/process.js index 355f73d542a6f979ef23dcc47be2b0f101f271da..87a8adbe541d58ccabc57c4e81b5238569e62327 100644 --- a/process/IndexSearch_lib/process.js +++ b/process/IndexSearch_lib/process.js @@ -1,61 +1,70 @@ -function IndexsearchUtils(){ - this.getAffectedIdValues = function(fieldname, affectedInfoContainer, updateFn){ - var affectedIds; - switch (affectedInfoContainer.action){ - case "I": - affectedIds = [affectedInfoContainer.newValues[affectedInfoContainer.columns.indexOf(fieldname)]]; - break; - case "U": - affectedIds = updateFn.call(null, affectedInfoContainer.id); - break; - case "D": - affectedIds = [affectedInfoContainer.oldValues[affectedInfoContainer.columns.indexOf(fieldname)]]; - break; - } - return affectedIds || []; - } - - this.createAffectedInfoContainer = function(changedIdValue, changedTable, action, columnsFn, oldValueFn, newValueFn){ - var res, internalStorage; - internalStorage = {}; - res = { - id: changedIdValue - ,table: changedTable - ,action: action - ,columns: null //null for autocomplete in the ADITO-designer - ,oldValues: null - ,newValues: null - }; - Object.defineProperty(res, "columns", { - get: function(){ - if (internalStorage["columns"] == undefined) - internalStorage["columns"] = columnsFn.call(null); - return internalStorage["columns"]; - } - ,set: function (v){ - internalStorage["columns"] = v; - } - }); - Object.defineProperty(res, "oldValues", { - get: function(){ - if (internalStorage["oldValues"] == undefined) - internalStorage["oldValues"] = oldValueFn.call(null); - return internalStorage["oldValues"]; - } - ,set: function (v){ - internalStorage["oldValues"] = v; - } - }); - Object.defineProperty(res, "newValues", { - get: function(){ - if (internalStorage["newValues"] == undefined) - internalStorage["newValues"] = newValueFn.call(null); - return internalStorage["newValues"]; - } - ,set: function (v){ - internalStorage["newValues"] = v; - } - }); - return res; - } +/** +* provides static methods for special handling of entities in JDito-Processes +* do not create an instance of this +* @class +* @static +*/ +function IndexsearchUtils(){ +} + +//todo: comment +IndexsearchUtils.getAffectedIdValues = function(fieldname, affectedInfoContainer, updateFn) { + var affectedIds; + switch (affectedInfoContainer.action){ + case "I": + affectedIds = [affectedInfoContainer.newValues[affectedInfoContainer.columns.indexOf(fieldname)]]; + break; + case "U": + affectedIds = updateFn.call(null, affectedInfoContainer.id); + break; + case "D": + affectedIds = [affectedInfoContainer.oldValues[affectedInfoContainer.columns.indexOf(fieldname)]]; + break; + } + return affectedIds || []; +} + +//todo: comment +IndexsearchUtils.createAffectedInfoContainer = function(changedIdValue, changedTable, action, columnsFn, oldValueFn, newValueFn) { + var res, internalStorage; + internalStorage = {}; + res = { + id: changedIdValue + ,table: changedTable + ,action: action + ,columns: null //null for autocomplete in the ADITO-designer + ,oldValues: null + ,newValues: null + }; + Object.defineProperty(res, "columns", { + get: function(){ + if (internalStorage["columns"] == undefined) + internalStorage["columns"] = columnsFn.call(null); + return internalStorage["columns"]; + } + ,set: function (v){ + internalStorage["columns"] = v; + } + }); + Object.defineProperty(res, "oldValues", { + get: function(){ + if (internalStorage["oldValues"] == undefined) + internalStorage["oldValues"] = oldValueFn.call(null); + return internalStorage["oldValues"]; + } + ,set: function (v){ + internalStorage["oldValues"] = v; + } + }); + Object.defineProperty(res, "newValues", { + get: function(){ + if (internalStorage["newValues"] == undefined) + internalStorage["newValues"] = newValueFn.call(null); + return internalStorage["newValues"]; + } + ,set: function (v){ + internalStorage["newValues"] = v; + } + }); + return res; } \ No newline at end of file diff --git a/process/Neon_lib/Neon_lib.aod b/process/Neon_lib/Neon_lib.aod new file mode 100644 index 0000000000000000000000000000000000000000..7cf15ed468aacf84f69010e2b03c1ad3ccf64e56 --- /dev/null +++ b/process/Neon_lib/Neon_lib.aod @@ -0,0 +1,6 @@ +<?xml version="1.0" encoding="UTF-8"?> +<process xmlns="http://www.adito.de/2018/ao/Model" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" VERSION="1.1.7" xsi:schemaLocation="http://www.adito.de/2018/ao/Model adito://models/xsd/process/1.1.7"> + <name>Neon_lib</name> + <majorModelMode>DISTRIBUTED</majorModelMode> + <process>%aditoprj%/process/Neon_lib/process.js</process> +</process> diff --git a/process/Neon_lib/process.js b/process/Neon_lib/process.js new file mode 100644 index 0000000000000000000000000000000000000000..441ffa8b7d4332333fd4d9af6dffad713d563e17 --- /dev/null +++ b/process/Neon_lib/process.js @@ -0,0 +1,384 @@ +/** + * Class containing utility functions for copying modules + * @class + */ +function CopyModuleUtils() { +} + +/** + * opens the new created modules in neonClient + * + * @param {String} pNeonContext req Name of the neon context that should be opened + * @param {Object} pModulesMapping req ModulesMapping object created by method copyModule + * + * @example var ModulesMapping = CMUtils.copyModule(InputMapping); + * + * CMUtils.openNewModules("Offer_context", ModulesMapping); + */ +CopyModuleUtils.openNewModules = function(pNeonContext, pModulesMapping) +{ + if(pModulesMapping != undefined) + { + var rootModule = Object.keys(pModulesMapping)[0]; + if(pModulesMapping[rootModule].DataRows != undefined) + { + var newids = []; + + for(var row in pModulesMapping[rootModule].DataRows) + { + newids.push(pModulesMapping[rootModule].DataRows[row].newPrimaryKey); + } + + if(newids.length > 0) + neon.openContext(pNeonContext, newids, neon.OPERATINGSTATE_VIEW, null); + } + } +} + +/** +* Creates a copy of a specified module together with specified subordinated modules. <br> +* The structure of the input mapping object is the following: <br> +* pInputMapping { +* (only one rootModule allowed) +* $rootModule$: { +* condition: "sqlWhereCondition" +* , ValueMapping: {$colName$: "value"} +* , destinationModuleName: "destinationModuleName" +* , DestinationColumnMapping: { $sourceColName$ : "destinationColName" } +* , SubModules: { +* $Module$:{ +* condition: "sqlWhereCondition" +* , ValueMapping: {$colName$: "value"} +* , destinationModuleName: "destinationModuleName" +* , DestinationColumnMapping: { $sourceColName$ : "destinationColName" } +* , SubModules: {...} +* } +* } +* } +* } +* +* @param {Object} pInputMapping +* +* @example var CMUtils = new CopyModuleUtils(); +* +* var InputMapping = { +* +* "OFFER": { +* condition: "OFFERID = '" + vars.get("$field.OFFERID") + "'" +* ,SubModules:{ +* "OFFERITEM": { +* condition: "OFFER_ID = '" + vars.get("$field.OFFERID") + "' order by ITEMSORT" +* } +* } +* } +* } +* +* CMUtils.copyModule(InputMapping); +*/ +CopyModuleUtils.copyModule = function(pInputMapping) +{ + var AliasDefinitionStructure = project.getAliasDefinitionStructure("Data_alias", null); + var ModulesMapping = {}; + var statements = []; + + buildMapping( pInputMapping ); + buildStatements( ModulesMapping ); + + if(statements.length > 0) + db.inserts( statements ); + + return ModulesMapping; + + /** + * Builds a mapping Object for the copyModule Method. <br> + * The structure of the Object is the following: <br> + * ModulesMapping = { + * + * $rootModule$: { ($$ marks an object property) + * + * (ModuleMapping) + * name: "moduleName" + * , destinationModuleName: "destinationModuleName" + * , DestinationColumnMapping: { $sourceColName$ : "destinationColName" } + * , ValueMapping: {$colName$: "value"} + * , dataRows:{ + * (ModuleRowMapping) + * $rowid$: { + * name: "moduleName" + * , oldPrimaryKey: "oldPrimaryKeyValue" + * , newPrimaryKey: "newPrimaryKeyValue" + * , ColumnMapping: { $colName$: { newValue: "newValue", oldValue: "oldValue", destinationColumn: "destinationColumn" } } + * , ModuleMapping: object reference to ModuleMapping object that contains ModuleRowMapping objects + * , ParentModuleMapping: object reference to supervised ModuleMapping object (at this point "null" because there is no supervised object) + * } + * } + * , SubModules: { + * (ModuleMapping) + * $moduleName$: { + * name: "moduleName" + * , destinationModuleName: "destinationModuleName" + * , DestinationColumnMapping: { $sourceColName$ : "destinationColName" } + * , ValueMapping: {$colName$: "value"} + * , dataRows:{ + * (ModuleRowMapping) + * $rowid$: { + * name: "moduleName" + * , oldPrimaryKey: "oldPrimaryKeyValue" + * , newPrimaryKey: "newPrimaryKeyValue" + * , ColumnMapping: { $colName$: { newValue: "newValue", oldValue: "oldValue", destinationColumn: "destinationColumn" } } + * , ModuleMapping: + * , ParentModuleMapping: object reference to supervised ModuleMapping object (at this point "null" because there is no supervised object) + * } + * } + * , SubModules: {...} + * } + * } + * + * } + * + *} + * + * @param {Object} pInputMapping InputMapping + */ + function buildMapping(pInputMapping) + { + //root mapping + var rootModule = Object.keys(pInputMapping)[0]; + var ModuleMapping = _ModuleMapping(rootModule, pInputMapping[rootModule]); + var ModuleData = _getModuleData(rootModule, pInputMapping[rootModule].condition); + + for(var row in ModuleData) + { + var ModuleRowMapping = _ModuleRowMapping(ModuleMapping, null, ModuleData[row]); + + ModuleMapping.DataRows[ModuleRowMapping.oldPrimaryKey] = ModuleRowMapping; + } + + ModulesMapping[rootModule] = ModuleMapping; + + //recursive subordinated modules mapping + _buildSubordinatedMapping(pInputMapping, ModuleMapping); + + + //delivers stored data for module in Database with condition + function _getModuleData(pModule, pCondition) + { + if(pModule == undefined) return {}; + + var ModuleColumnsStructure = AliasDefinitionStructure.tables[pModule].columns; + var cols = Object.keys(ModuleColumnsStructure); + + var condition = "1=1"; + if(pCondition != undefined) + condition = pCondition; + + var dbData = db.table("select " + cols.join(", ") + " from " + pModule + " where " + condition); + + //map 2d-Array to Object { $rowNumber$: { $columnName$: { value: "valueInDB" } } } + var DataObj = {}; + for(var row = 0; row < dbData.length; row++) + { + DataObj[row] = {}; + for(var col = 0; col < dbData[row].length; col++) + { + DataObj[row][cols[col]] = { + value: dbData[row][col] + }; + } + } + + return DataObj; + } + + + //recursive: ModuleMapping and ModuleRowMapping for subordinated modules + function _buildSubordinatedMapping(pInputMapping, pParentModuleMapping) + { + var SubModules = pInputMapping[pParentModuleMapping.name].SubModules; + if(SubModules == undefined) + return; + + for(var subModuleName in SubModules) + { + var ModuleMapping = _ModuleMapping(subModuleName, SubModules[subModuleName]); + ModuleData = _getModuleData(subModuleName, SubModules[subModuleName].condition); + for(var row in ModuleData) + { + ModuleRowMapping = _ModuleRowMapping(ModuleMapping, pParentModuleMapping, ModuleData[row]); + ModuleMapping.DataRows[ModuleRowMapping.oldPrimaryKey] = ModuleRowMapping; + } + + ModulesMapping[pParentModuleMapping.name].SubModules[subModuleName] = ModuleMapping; + + _buildSubordinatedMapping(SubModules, ModuleMapping); + } + } + + function _ModuleMapping( pModuleName, pInputModuleMapping ) + { + return { + name: pModuleName, + destinationModuleName: pInputModuleMapping.destinationModuleName == undefined ? pModuleName : pInputModuleMapping.destinationModuleName, + DestinationColumnMapping: pInputModuleMapping.DestinationColumnMapping == undefined ? {} : pInputModuleMapping.DestinationColumnMapping, + ValueMapping: pInputModuleMapping.ValueMapping == undefined ? {} : pInputModuleMapping.ValueMapping, + DataRows:{}, + SubModules: {} + }; + } + + function _ModuleRowMapping(pModuleMapping, pParentModuleMapping, pDataRow) + { + var ModuleRowMapping = { + name: pModuleMapping.name, + oldPrimaryKey: null, + newPrimaryKey: null, + ColumnMapping: {}, + ModuleMapping: pModuleMapping, + ParentModuleMapping: pParentModuleMapping + }; + + var ModuleColumnsStructure = AliasDefinitionStructure.tables[ModuleRowMapping.name].columns; + + //build ColumnMapping + for(var col in ModuleColumnsStructure) + { + //set defined columns from InputMapping -> if not defined, use the same column + var destinationColumn = ModuleRowMapping.ModuleMapping.DestinationColumnMapping[col]; + if(destinationColumn == undefined) + destinationColumn = col; + + //set defined values from InputMapping -> if not defined, use the value from DB + var oldValue = pDataRow[col].value; + var newValue = newValue = ModuleRowMapping.ModuleMapping.ValueMapping[col]; + if(newValue == undefined) + newValue = oldValue; + + //set new primary key + if(ModuleColumnsStructure[col].primaryKey) + { + ModuleRowMapping.oldPrimaryKey = ModuleRowMapping.ColumnMapping[col] = oldValue; + newValue = util.getNewUUID(); + ModuleRowMapping.newPrimaryKey = newValue; + } + + ModuleRowMapping.ColumnMapping[col] = _columnMapping(newValue, oldValue, destinationColumn); + } + + switch(ModuleRowMapping.name) + { + case "OFFER": + { + //andere Values setzen + ModuleRowMapping.ColumnMapping["OFFERDATE"].newValue = DateUtils.getTodayUTC(); + } + break; + case "OFFERITEM": + { + //OFFER_ID mappen + if(ModuleRowMapping.ParentModuleMapping.name == "OFFER") + { + ModuleRowMapping.ColumnMapping["OFFER_ID"].newValue = ModulesMapping[ModuleRowMapping.ParentModuleMapping.name].DataRows[ModuleRowMapping.ColumnMapping["OFFER_ID"].oldValue].newPrimaryKey; + } + //ASSIGNEDTO mappen + if(ModuleRowMapping.ColumnMapping["ASSIGNEDTO"].oldValue != "") + { + if(ModuleRowMapping.ParentModuleMapping == null) + { + ModuleRowMapping.ColumnMapping["ASSIGNEDTO"].newValue = ModulesMapping["OFFERITEM"].DataRows[ModuleRowMapping.ColumnMapping["ASSIGNEDTO"].oldValue].newPrimaryKey; + } + else + { + ModuleRowMapping.ColumnMapping["ASSIGNEDTO"].newValue = ModuleRowMapping.ModuleMapping.DataRows[ModuleRowMapping.ColumnMapping["ASSIGNEDTO"].oldValue].newPrimaryKey; + } + } + } + break; + case "OFFERLINK": + { + + } + default: + { + + } + + + } + + //Spezialbehandlung USER_NEW DATENEW.... + if(ModuleRowMapping.ColumnMapping["DATE_NEW"] != undefined) ModuleRowMapping.ColumnMapping["DATE_NEW"].newValue = datetime.date(); + if(ModuleRowMapping.ColumnMapping["USER_NEW"] != undefined) ModuleRowMapping.ColumnMapping["USER_NEW"].newValue = vars.get("$sys.user"); + if(ModuleRowMapping.ColumnMapping["DATE_EDIT"] != undefined) ModuleRowMapping.ColumnMapping["DATE_EDIT"].newValue = ""; + if(ModuleRowMapping.ColumnMapping["USER_EDIT"] != undefined) ModuleRowMapping.ColumnMapping["USER_EDIT"].newValue = ""; + + return ModuleRowMapping; + } + + + + + function _columnMapping(pNewValue, pOldValue, pDestinationColumn) + { + return { + newValue: pNewValue, + oldValue: pOldValue, + destinationColumn: pDestinationColumn + }; + } + + } + + /** + * Builds the insert statements for passed ModulesMapping + * + * @param {Object} pModulesMapping ModulesMapping from buildMapping() + */ + function buildStatements(pModulesMapping) + { + var rootModule = Object.keys(pModulesMapping)[0]; + + for(var row in pModulesMapping[rootModule].DataRows) + { + //buildInsertStatement + statements.push(_statement(pModulesMapping[rootModule].DataRows[row])); + } + + _subordinatedStatements(pModulesMapping[rootModule]); + + function _subordinatedStatements(pMapping) + { + if(pMapping.SubModules == undefined) + return; + + for(var subModule in pMapping.SubModules) + { + + for(var row in pMapping.SubModules[subModule].DataRows) + { + statements.push(_statement(pMapping.SubModules[subModule].DataRows[row])); + } + + _subordinatedStatements(pMapping.SubModules[subModule]); + } + + } + + function _statement(pRowMapping) + { + var cols = []; + var vals = []; + var destTable = pRowMapping.ModuleMapping.destinationModuleName; + var colMapping = pRowMapping.ColumnMapping; + + for(var col in colMapping) + { + cols.push(colMapping[col].destinationColumn); + vals.push(colMapping[col].newValue.toString()); + } + + var colTypes = db.getColumnTypes(destTable, cols) + + return [destTable, cols, colTypes, vals]; + } + } +} diff --git a/process/OfferOrder_lib/process.js b/process/OfferOrder_lib/process.js index 3e32be2ad1ab81fd1382f913485c231788c09444..7ace6bd3b1719873563408cbf29b80f968d9ac61 100644 --- a/process/OfferOrder_lib/process.js +++ b/process/OfferOrder_lib/process.js @@ -15,8 +15,7 @@ function OfferUtils() */ this.getNextOfferNumber = function() { - var JdUtils = new JDitoUtils(); - return JdUtils.getNextUniqueNumber("OFFERCODE", "OFFER"); + return NumberSequencingUtils.getNextUniqueNumber("OFFERCODE", "OFFER"); } /** @@ -26,8 +25,7 @@ function OfferUtils() */ this.getNextOfferVersionNumber = function(pOfferCode) { - var JdUtils = new JDitoUtils(); - return JdUtils.getNextUniqueNumber("VERSNR", "OFFER", 1, "OFFERCODE = " + pOfferCode); + return NumberSequencingUtils.getNextUniqueNumber("VERSNR", "OFFER", 1, "OFFERCODE = " + pOfferCode); } /** @@ -39,8 +37,7 @@ function OfferUtils() */ this.validateOfferNumber = function(pOfferNumber) { - var JdUtils = new JDitoUtils(); - return JdUtils.validateUniqueNumber(pOfferNumber, "OFFERCODE", "OFFER"); + return NumberSequencingUtils.validateUniqueNumber(pOfferNumber, "OFFERCODE", "OFFER"); } this.getOfferNumberValidationFailString = function() diff --git a/process/Product_lib/process.js b/process/Product_lib/process.js index 161c665d32d38340cba27e1bb437dd318b2b30b8..94451a5e126ea8427c36a2cdf0a633d5633e6bd8 100644 --- a/process/Product_lib/process.js +++ b/process/Product_lib/process.js @@ -82,7 +82,7 @@ function ProductUtils() * @param pPriceListFilter {Object} opt { currency: "currencyValue", quantity: "quantityValue", relationId: "relationIdValue (for custom price lists)" } * * @example //Product_entity, Field: PRODUCT_ID, Process: onValueChange - * var pid = ProcessHandlingUtil.getOnValidationValue(vars.get("$field.PRODUCT_ID")); + * var pid = ProcessHandlingUtils.getOnValidationValue(vars.get("$field.PRODUCT_ID")); * var curr = vars.exists("$param.Currency_param") ? vars.get("$param.Currency_param") : ""; * var relid = vars.exists("$param.RelationId_param") ? vars.get("$param.RelationId_param") : ""; * var pUtils = new ProductUtils(); @@ -280,7 +280,7 @@ function ProductUtils() * @example //Productprice_entity, Field: PRICELIST, Process: onValidation * var pUtils = new ProductUtils(); * var priceList = { - * priceList: ProcessHandlingUtil.getOnValidationValue(vars.get("$field.PRICELIST")) + * priceList: ProcessHandlingUtils.getOnValidationValue(vars.get("$field.PRICELIST")) * , fromQuantity: vars.get("$field.FROMQUANTITY") * , buySell: vars.get("$field.BUYSELL") * , currency: vars.get("$field.CURRENCY") diff --git a/process/Salesproject_lib/process.js b/process/Salesproject_lib/process.js index e3f10b6a49e5053b051b02a90dadc40c94dcf995..231a2538d20894fff9b6f9774568d282a3721d1c 100644 --- a/process/Salesproject_lib/process.js +++ b/process/Salesproject_lib/process.js @@ -24,8 +24,7 @@ Salesproject = { * @result {String} next valid project number */ getNextProjectNumber: function() { - var JdUtils = new JDitoUtils(); - return JdUtils.getNextUniqueNumber("PROJECTCODE", "SALESPROJECT"); + return NumberSequencingUtils().getNextUniqueNumber("PROJECTCODE", "SALESPROJECT"); }, /** @@ -36,8 +35,7 @@ Salesproject = { * @result {boolean} passed number is valid */ validateProjectNumber: function(projectNumber) { - var JdUtils = new JDitoUtils(); - return JdUtils.validateUniqueNumber(projectNumber, "PROJECTCODE", "SALESPROJECT"); + return NumberSequencingUtils.validateUniqueNumber(projectNumber, "PROJECTCODE", "SALESPROJECT"); }, /** diff --git a/process/Sql_lib/process.js b/process/Sql_lib/process.js index 61a7c941e309be1fc8a8673b61865554c633b333..15ba49d93fcc1bca95135476814c1f5798f85b0e 100644 --- a/process/Sql_lib/process.js +++ b/process/Sql_lib/process.js @@ -657,6 +657,7 @@ SqlUtils.getSingleColumnType = function(fieldOrTableName, columnName, alias) { /** *Class containing utilities for SQL *@deprecated use SqlMaskingUtils + *@todo: shift to newer objects or remove *@class */ function LegacySqlUtils() @@ -1169,55 +1170,7 @@ function LegacySqlUtils() return res; } - /** - * Setzt eine Condition zusammen und liefert sie zurück - * builds a conditions and returns it - * - * @param {Object} pValue req Filtervalue - * @param {String} pCondition req variable in which the condition should be written - * @param {String} pWhere req additional condition - * @param {Integer} pSQLType opt SQLTYPES type of pValue - * @param {Array} pPreparedValues opt Value for the condition, if it's a prepared statement - * - * @return {String} - */ - this.makeCondition = function( pValue, pCondition, pWhere, pSQLType, pPreparedValues) - { - if ( pValue != "" ) - { - if ( pCondition != "" ) - pCondition += " and "; - - pCondition += pWhere; - if(pPreparedValues != undefined) - { - pPreparedValues.push([pValue, pSQLType]); - } - } - return pCondition; - } - /** - * returns a type of column in the database - * - * @param {String} pTableName req name of a table (e.g. "EVENT") OR if pColumnName is not passed table.column (e.g. "EVENT.STATUS") - * @param {String} pColumnName opt name of column (e.g. "STATUS") if in pTableName only tablename is passed - * @param {String} pAlias opt Alias to the database where the type should be loaded; default is current alias - * - * @return {String} type of column such as SQLTYPES.xyz - */ - this.getSingleColumnType = function(pTableName, pColumnName, pAlias) - { - if (pColumnName == undefined) - { - pColumnName = pTableName.substring(pTableName.indexOf(".") + 1); - pTableName = pTableName.substring(0, pTableName.indexOf(".")); - } - if (pAlias == undefined) - pAlias = db.getCurrentAlias(); - - return db.getColumnTypes(pTableName, [pColumnName], pAlias)[0]; - } /** * calls a given function for N blocks of sql-data as long as records are available or the paging-process is manually canceled * diff --git a/process/Util_lib/process.js b/process/Util_lib/process.js index 969fe5b2f57169f259666d6e1a2e6f9059d1b40a..ace960b642dbf072b869738f86de3d5a4f9a4bc1 100644 --- a/process/Util_lib/process.js +++ b/process/Util_lib/process.js @@ -15,819 +15,407 @@ import("OfferOrder_lib"); import("Date_lib"); /** - * Class containing utility functions for images - * + * Class containing static utility functions for use with arrays * @class - * + * @static */ -function ImageUtils() -{ - var that = this; - /** - * reads a picture depending on description and icon_type from ASYS_ICONS - * @param {String} pIconType requ ICON_TYPE of ASYS_ICONS - * @param {String} pDescription requ DESCRIPTION of ASYS_ICONS - * @param {Boolean} pBase64 opt true/undefined: BINDATA (Tables, trees) or false: ID (Icons, XMPP picture) - * - * @return {String} BINDATA of ASYS_ICONS - * - * - */ - this.getIcon = function(pIconType, pDescription, pBase64) - { - var field = pBase64 || pBase64 == undefined ? "BINDATA" : "ID"; - return db.cell("select " + field + " from ASYS_ICONS where DESCRIPTION = '" + pDescription + "' and ICON_TYPE = '" + pIconType + "'", - "_____SYSTEMALIAS"); - } - - /** - * reads frequently used combination of description and bindata/id from ASYS_ICONS - * @param {String} pIconType ICON_TYPE of ASYS_ICONS - * @param {Boolean} pBase64 opt true: BINDATA or false/undefined: ID - * - * @return {String[][]} Key: Description, Value: ID - * - */ - this.getImageObject = function (pIconType, pBase64) - { - var field = pBase64 ? "BINDATA" : "ID"; - - var icons = {};//kein Array sondern Object, da es sich um ein assoziatives Array handelt - var data = db.table("select " + field + ", DESCRIPTION from ASYS_ICONS where ICON_TYPE = '" + pIconType + "'", - "_____SYSTEMALIAS"); - for (var i = 0; i < data.length; i++ ) - icons[data[i][1]] = data[i][0]; - - return icons; - } +function ArrayUtils(){ } -/** - * class containing utils for URLs - * @class - */ -function UrlUtils() -{ - var that = this; - /** - * Extracts a query param from an URL - * - * @param {String} pURL the text from which the parameter should be read - * @param {String} pParameter Name of the parameter - * (Messaging mit Default Trenner: - * - im.internal: "id=", "title_ack=", "title_dcl=", "text_ack=", "text_dcl=", "desc=", "name=", "result=", "decidable=", "icon=", "icon_ack=", "icon_dcl=" - * - im.file: "id=", "size=", "result=", "requestor=") - * @param {String} pSplit opt der String, which marks the end of the param value, standard is '&' - * - * @return {String} the extracted value or null if param not existent - * - * - */ - this.extractBlock = function(pURL, pParameter, pSplit) - { - var idLeft = pURL.indexOf(pParameter, 0); - if(idLeft <= -1) - return null; // Kein Block vorhanden - if(pSplit != undefined) var idRight = pURL.indexOf(pSplit, idLeft); - if(pSplit == undefined || idRight <= -1) idRight = pURL.indexOf("&", idLeft); - if(idRight <= -1) - idRight = pURL.length; - - return pURL.substring(idLeft + pParameter.length, idRight); - } - - /** - * opens the browser - * - * @param {String} pUrl req - * @param {String} pBrowserType opt Browser type 1 = Native (Internet Explorer Rendering Engine) - * ,2 = JavaFX (WebKit Rendering Engine), 3 = Native, if possible. - * default is 2; - * - * @return {void} - * - * - */ - this.openUrl = function(pUrl, pBrowserType) - { - if ( pUrl != "" ) - { - if (pUrl.substr(0, 4) != "http" - && pUrl.substr(0, 4) != "ftp:" - && pUrl.substr(0, 5) != "file:") - { - pUrl = "http://" + pUrl; - } - var browserType = pBrowserType || 2; - swing.openBrowser(true, false, encodeURI(pUrl), swing.WINDOW_CURRENT, [], browserType); - } - } -} /** - * Class containing String utility functions - * @class + * sorts a two dimensional array by the given index + * + * @param {Array} targetArray the Array to be sorted + * @param {String} index the index of the field to sort by + * @param {Boolean} [sortAsc=false] TRUE sorts ascending, FALSE sorts decending + * @param {Boolean} [isNumber=false] TRUE sorts numerical, FALSE or undefined sorts alphanumerical + * + * @return targetArray + * + * @throws {RangeError} if index is outside pArray.length + * * */ -function StringUtils() -{ - var that = this; - /** - * validates if a string is a UUID according to versions 1 through 5 of RFC4122 - * @param {String} pUUID String to be validated - * @param {bool} pIgnoreCase true if case sensitivty isn't needed - * - * @return {Boolean} ob es sich um eine UUID handelt - * - * - */ - this.isValidUUID = function(pUUID, pIgnoreCase) - { - if (pIgnoreCase) - { - pUUID = pUUID.toLowerCase(); +ArrayUtils.sort2d = function(targetArray, index, sortAsc, isNumber){ + if (targetArray.length == 0) + return targetArray; + if (targetArray[0].length == 0) + return targetArray; + if (index >= targetArray[0].length) + throw new RangeError("Index Out Of Bounds: " + index + " >= " + targetArray[0].length); + + var sortFn = function (x, y){ + if ( isNumber ) { + xx = Number(x[index]); + yy = Number(y[index]); } - - return new RegExp(/^[0-9a-f]{8}-[0-9a-f]{4}-[1-5][0-9a-f]{3}-[89ab][0-9a-f]{3}-[0-9a-f]{12}$/).test(pUUID); - } - /** - * Returns wether a string starts with given string or not. - * @param pString String, which is being searched in - * @param pStringToSearch String, which is being searched - * - * @return {Boolean} - * - * - */ - this.stringStatrsWith = function(pString, pStringToSearch) - { - return pString.indexOf(pStringToSearch) === 0; - } - /** - * Splits street and buildingnumber - * - * @param {String} pAddress req Address - * - * @return {Array} [ Street, BuildingNr ] - * - * - */ - this.splitAddress = function(pAddress) - { - var ret = ["",""]; - if(pAddress != "") - { - var arr = pAddress.match( /^[^0-9]+|[0-9]+.*$/g); - ret[0] = arr[0].trim(); - if(arr[1]) - ret[1] = arr[1].trim(); + else { + xx = x[index]; + yy = y[index]; + + xx = xx.toLowerCase(); + xx = xx.replace(/ä/g,"ae"); + xx = xx.replace(/ö/g,"oe"); + xx = xx.replace(/ü/g,"ue"); + xx = xx.replace(/ß/g,"ss"); + + yy = yy.toLowerCase(); + yy = yy.replace(/ä/g,"ae"); + yy = yy.replace(/ö/g,"oe"); + yy = yy.replace(/ü/g,"ue"); + yy = yy.replace(/ß/g,"ss"); } - return ret; + if (xx == yy) + return 0; + if (xx < yy) + return (sortAsc ? -1 : 1); + return (sortAsc ? 1 : -1); } - /** - * Generates the string which shows, who created the dataset and who edited it - * - * @param {String} pDateNew req Date of creation - * @param {String} pUserNew req name of the user, who created the dataset - * @param {String} pDateEdit req date of the last change - * @param {String} pUserEdit req name of last editor - * - * @return {String} - * - * - */ - this.label_new_edit = function(pDateNew, pUserNew, pDateEdit, pUserEdit) - { - var retst = translate.withArguments("created on %0 by %1", [ datetime.toDate(pDateNew, translate.text("yyyy-MM-dd")), pUserNew]); - if (pDateEdit != undefined && pDateEdit != "") retst += " | " + translate.withArguments("changed on %0 by %1", [ datetime.toDate(pDateEdit, translate.text("yyyy-MM-dd")), pUserEdit]); - return(retst) - } - - /** - * uses the right translate method, depending on the parameters - * - * @param {String} pText string to be translated - * @param {String} pLocale locale for translating - * - * @return {String} - * - * - */ - this.translateStr = function( pText, pLocale ) - { - if ( pLocale == undefined ) - return translate.text(pText); - else - return translate.text(pText, pLocale) - } - - + targetArray.sort(sortFn); + return targetArray } -/** - * Class containing utility functions for use with arrays - * @class - */ -function DataUtils() -{ - var that = this; - /** - * sorts a two dim array by the given index - * - * @param {Array} pArray req the Array to be sorted - * @param {String} pIndex req the index of the field to sort by - * @param {Boolean} pUp req TRUE sorts ascending, FALSE sorts decending - * @param {Boolean} isNumber TRUE sorts numerical, FALSE or undefined sorts alphanumerical - * - * @return void - * - * @throws {Error} if index is outside pArray.length - * - * - */ - this.array_mDimSort = function(pArray, pIndex, pUp, isNumber) - { - if (pArray.length == 0) - return; - - if (pArray[0].length == 0) - return; - if (pIndex >= pArray[0].length) - throw new Error("Index Out Of Bounds: " + pIndex + " >= " + pArray[0].length); - - /* - * the sort function - * - * @param {String} x req value 1 - * @param {String} y req value 2 - * - * @return {Integer} - * - * @memberof array_mDimSort - */ - function array_mDimSorter(x, y) - { - if ( isNumber ) - { - xx = Number(x[pIndex]); - yy = Number(y[pIndex]); - } - else - { - xx = x[pIndex]; - yy = y[pIndex]; - - //Umlaute austauschen - xx = xx.toLowerCase(); - xx = xx.replace(/ä/g,"ae"); - xx = xx.replace(/ö/g,"oe"); - xx = xx.replace(/ü/g,"ue"); - xx = xx.replace(/ß/g,"ss"); - - yy = yy.toLowerCase(); - yy = yy.replace(/ä/g,"ae"); - yy = yy.replace(/ö/g,"oe"); - yy = yy.replace(/ü/g,"ue"); - yy = yy.replace(/ß/g,"ss"); - } - if (xx == yy) - return 0; - - if (xx < yy) - return (pUp ? -1 : 1); - - return (pUp ? 1 : -1); +/** +* sorts an array up to 6 columns with sortorder +* +* @param {Array} targetArray req the array with data +* @param {Integer} us req the Sortorder for Column 1 = Param u (1=asc, -1=desc) +* @param {Integer} u req the 1 Column +* @param {Integer} vs opt the Sortorder for Column 2 = Param v (1=asc, -1=desc) +* @param {Integer} v opt the 2 Column +* @param {Integer} ws opt the Sortorder for Column 3 = Param w (1=asc, -1=desc) +* @param {Integer} w opt the 3 Column +* @param {Integer} xs opt the Sortorder for Column 4 = Param x (1=asc, -1=desc) +* @param {Integer} x opt the 4 Column +* @param {Integer} ys opt the Sortorder for Column 5 = Param y (1=asc, -1=desc) +* @param {Integer} y opt the 5 Column +* @param {Integer} zs opt the Sortorder for Column 6 = Param z (1=asc, -1=desc) +* @param {Integer} z opt the 6 Column +* +* @return {void} +*/ +ArrayUtils.sortMulti = function(targetArray, us, u, vs, v, ws, w, xs, x, ys, y, zs, z) { + /* + * sort of a two dim array, up to 6 columns + * + * @param {String} a req value 1, first compared element + * @param {String} b req value 2, sencond compared element + * + * @return {Integer} -1 - set a below b, 0 - equal, 1 - set b below a + */ + var sortFn = function(a, b) { + var stringComparison = function(a, b) { + a = a.toLowerCase(); + a = a.replace(/ä/g,"ae"); + a = a.replace(/ö/g,"oe"); + a = a.replace(/ü/g,"ue"); + a = a.replace(/ß/g,"ss"); + + b = b.toLowerCase(); + b = b.replace(/ä/g,"ae"); + b = b.replace(/ö/g,"oe"); + b = b.replace(/ü/g,"ue"); + b = b.replace(/ß/g,"ss"); + + return( a == b ) ? 0 : ( a > b ) ? 1 : -1; } - pArray.sort(array_mDimSorter); - } - - /** - * sorts an array up to 6 columns with sortorder - * - * @param {Array} pArray req the array with data - * @param {Integer} us req the Sortorder for Column 1 = Param u (1=asc, -1=desc) - * @param {Integer} u req the 1 Column - * @param {Integer} vs opt the Sortorder for Column 2 = Param v (1=asc, -1=desc) - * @param {Integer} v opt the 2 Column - * @param {Integer} ws opt the Sortorder for Column 3 = Param w (1=asc, -1=desc) - * @param {Integer} w opt the 3 Column - * @param {Integer} xs opt the Sortorder for Column 4 = Param x (1=asc, -1=desc) - * @param {Integer} x opt the 4 Column - * @param {Integer} ys opt the Sortorder for Column 5 = Param y (1=asc, -1=desc) - * @param {Integer} y opt the 5 Column - * @param {Integer} zs opt the Sortorder for Column 6 = Param z (1=asc, -1=desc) - * @param {Integer} z opt the 6 Column - * - * @return {void} - * - * - */ - this.sortArray = function(pArray, us, u, vs, v, ws, w, xs, x, ys, y, zs, z) - { - pArray.sort(_sortmulti); - /* - * sort of a two dim array, up to 6 columns - * - * @param {String} a req value 1, first compared element - * @param {String} b req value 2, sencond compared element - * - * @return {Integer} -1 - set a below b, 0 - equal, 1 - set b below a - */ - function _sortmulti(a, b) - { - var stringComparison = function(a, b) - { - a = a.toLowerCase(); - a = a.replace(/ä/g,"ae"); - a = a.replace(/ö/g,"oe"); - a = a.replace(/ü/g,"ue"); - a = a.replace(/ß/g,"ss"); - - b = b.toLowerCase(); - b = b.replace(/ä/g,"ae"); - b = b.replace(/ö/g,"oe"); - b = b.replace(/ü/g,"ue"); - b = b.replace(/ß/g,"ss"); - return( a == b ) ? 0 : ( a > b ) ? 1 : -1; - } - - - var swap=0; + var swap=0; - if (isNaN(a[u] - b[u])) // if there is a string in the first compared element - if( isNaN(a[u]) && isNaN(b[u]) ) // if both are strings, - swap = stringComparison(a[u], b[u]); // then: true - false = 1; false - true = -1 - else - swap = (isNaN(a[u]) ? 1 : -1); + if (isNaN(a[u] - b[u])) // if there is a string in the first compared element + if( isNaN(a[u]) && isNaN(b[u]) ) // if both are strings, + swap = stringComparison(a[u], b[u]); // then: true - false = 1; false - true = -1 else - swap = (a[u] - b[u]); + swap = (isNaN(a[u]) ? 1 : -1); + else + swap = (a[u] - b[u]); - if ((v == undefined) || (swap != 0)) - return swap * us; - else - if (isNaN(a[v] - b[v])) - if ((isNaN(a[v])) && (isNaN(b[v]))) - swap = stringComparison(a[v], b[v]); - else - swap = (isNaN(a[v]) ? 1 : -1); + if ((v == undefined) || (swap != 0)) + return swap * us; + else + if (isNaN(a[v] - b[v])) + if ((isNaN(a[v])) && (isNaN(b[v]))) + swap = stringComparison(a[v], b[v]); else - swap = (a[v] - b[v]); + swap = (isNaN(a[v]) ? 1 : -1); + else + swap = (a[v] - b[v]); - if ((w == undefined) || (swap != 0)) - return swap * vs; - else - if (isNaN(a[w] - b[w])) - if ((isNaN(a[w])) && (isNaN(b[w]))) - swap = stringComparison(a[w], b[w]); - else - swap = (isNaN(a[w]) ? 1 : -1); + if ((w == undefined) || (swap != 0)) + return swap * vs; + else + if (isNaN(a[w] - b[w])) + if ((isNaN(a[w])) && (isNaN(b[w]))) + swap = stringComparison(a[w], b[w]); else - swap = (a[w] - b[w]); + swap = (isNaN(a[w]) ? 1 : -1); + else + swap = (a[w] - b[w]); - if ((x == undefined) || (swap != 0)) - return swap * ws; - else - if (isNaN(a[x] - b[x])) - if ((isNaN(a[x])) && (isNaN(b[x]))) - swap = stringComparison(a[x], b[x]); - else - swap = (isNaN(a[x]) ? 1 : -1); + if ((x == undefined) || (swap != 0)) + return swap * ws; + else + if (isNaN(a[x] - b[x])) + if ((isNaN(a[x])) && (isNaN(b[x]))) + swap = stringComparison(a[x], b[x]); else - swap = (a[x] - b[x]); + swap = (isNaN(a[x]) ? 1 : -1); + else + swap = (a[x] - b[x]); - if ((y == undefined) || (swap != 0)) - return swap * xs; - else - if (isNaN(a[y] - b[y])) - if ((isNaN(a[y])) && (isNaN(b[y]))) - swap = stringComparison(a[y], b[y]); - else - swap = (isNaN(a[y]) ? 1 : -1); + if ((y == undefined) || (swap != 0)) + return swap * xs; + else + if (isNaN(a[y] - b[y])) + if ((isNaN(a[y])) && (isNaN(b[y]))) + swap = stringComparison(a[y], b[y]); else - swap = (a[y] - b[y]); + swap = (isNaN(a[y]) ? 1 : -1); + else + swap = (a[y] - b[y]); - if ((z == undefined) || (swap != 0)) - return swap * ys; - else - if(isNaN(a[z] - b[z])) - if((isNaN(a[z])) && (isNaN(b[z]))) - swap = stringComparison(a[z], b[z]); - else - swap = (isNaN(a[z]) ? 1 : -1); + if ((z == undefined) || (swap != 0)) + return swap * ys; + else + if(isNaN(a[z] - b[z])) + if((isNaN(a[z])) && (isNaN(b[z]))) + swap = stringComparison(a[z], b[z]); else - swap = (a[z] - b[z]); - - return swap * zs; - } - - } - - /** - * removes an element from an array - * - * @param {Array} pArray req Array from which the element should be removed - * @param {String} pElement req index of the element which should be removed - * - * @return {Array} array without the removed element - * - * - */ - this.removeElement = function(pArray, pElement) - { - return pArray.splice(pElement, 1); - } - - /** - * returns all elemtens not in pSource - * - * @param {Array} pSource req this elements are being searched - * @param {Array} pReference req in this array the search is done - * @param {Boolean} pIgnoreCase opt - * - * @return {Array[]} resultIn - * - * - */ - this.notIn = function(pSource, pReference, pIgnoreCase) - { - var resultIn = new Array(); - for (var i = 0; i < pSource.length; i++) - { - if (!that.hasElement(pReference, pSource[i], pIgnoreCase)) - resultIn.push(pSource[i]); - } - - return resultIn; - } - - /** - * returns if an element is in an array. - * - * @param {Array} pArray req array which should be searched in - * @param {String} pElement req Element which should looked for - * @param {Boolean} pIgnoreCase opt - * - * @return {Boolean} true if it has the element - * - * - */ - this.hasElement = function(pArray, pElement, pIgnoreCase) - { - for (var i = 0; i < pArray.length; i++) - { - if ( pIgnoreCase != undefined && pIgnoreCase ) - { - if (pArray[i].toLowerCase() == pElement.toLowerCase()) - return true; - } - else - if (pArray[i] == pElement) - return true; - } + swap = (isNaN(a[z]) ? 1 : -1); + else + swap = (a[z] - b[z]); - return false; + return swap * zs; } - - /** - * concats arrays column by column - * see example for more details - * logging.show(JSON.stringify(concatArrayColumns(a, b))); - * //[["a","A"],["b","B"],["c","C"]] - * - * logging.show(JSON.stringify(a.concat(b))); - * //["a","b","c","A","B","C"] - * @param {Array} pArray1 req you have to pass at least 2 Arrays that shall be concated but you can pass as many as you want - * @param {Array} pArrayN req you have to pass at least 2 Arrays that shall be concated but you can pass as many as you want - * - * - * - * @return {Array} Beschreibung - * - * - */ - this.concatArrayColumns = function(pArray1, pArrayN) - { - var res = []; - for (var i = 0, l = arguments.length; i < l; i++)//this function can handle an "unlimited" amount of functionparams - { - var inpArr = arguments[i]; - for (var ii = 0, ll = inpArr.length; ii < ll; ii++) - { - if (res[ii] == undefined) - res[ii] = []; - - res[ii] = res[ii].concat(inpArr[ii]); - } - } - return res; - } - - /** - * function to determine, if an object is truly an Array - * @param {Object} pArray - * @return {Boolean} returns wether an Object is an Array (true) or not (false) - * - * - */ - this.isArray = function(pArray) - { - return (pArray && typeof pArray === "object" && typeof pArray.length === "number" && !(pArray.propertyIsEnumerable('length'))) - } + targetArray.sort(sortFn); + return targetArray; } -/** - * Class containing utility functions for table components - * @class - */ -function TableUtils() -{ - var that = this; - /** - * Moving Tablerows Sort field up or down a step - * - * @param {String} pTable req Table - * @param {String} pTableComp req name of the table comp - * @param {String} pCountField req sort field name z.B. 'SORT' - * @param {String} pDirection req 'up' oder 'down' - * @param {String} pCondition opt - * - * @return {void} - * - * - */ - this.moveRow = function(pTable, pTableComp, pCountField, pDirection, pCondition) - { - that.sortRow(pTable, pCountField, pCondition); - - if (pCondition == undefined || pCondition == "") - pCondition = ""; - else - pCondition = " and " + pCondition; - - var id = text.decodeFirst(vars.getString(pTableComp)); - var col = [pCountField]; - var nextval; - var typ = db.getColumnTypes( pTable, col); - var oldval = db.cell("select " + pCountField + " from " + pTable + " where " + pTable + "ID = '" + id + "'"); - - if (pDirection == "up") - { - nextval = db.cell("select max(" + pCountField + ") from " + pTable - + " where " + pCountField + " < " + oldval + pCondition); - } - else - { - nextval = db.cell("select min(" + pCountField + ") from " + pTable - + " where " + pCountField + " > " + oldval + pCondition); - } - if ( nextval == "" ) - nextval = 0; - var nextID = db.cell("select " + pTable + "ID from " + pTable + " where " + pCountField + " = " + nextval + pCondition); - db.updateData(pTable, col, typ, [oldval], pTable + "ID = '" + nextID + "'") - db.updateData(pTable, col, typ, [nextval], pTable + "ID = '" + id + "'") - } - /** - * activates Up/Down buttons - * - * @param {String} pTable req Table - * @param {String} pTableComp req name of the table comp - * @param {String} pSortfield req sort field name z.B. 'SORT' - * @param {String} pDirection req 'up' or 'down' - * @param {String} pCondition opt - * - * @return {Boolean} true if active - * - * - */ - this.moveActive = function(pTable, pTableComp, pSortfield, pDirection, pCondition) - { - var oldval = "min"; - var minmax = "min"; - var id = text.decodeFirst(vars.getString(pTableComp)); - - if (pCondition == undefined || pCondition == "") - pCondition = ""; - else - pCondition = " where " + pCondition; - - if (id != "" && swing.getTableDataByAttribute(pTableComp, swing.INSERTED ).length == 0 && vars.getString("$sys.workingmode") == swing.FRAMEMODE_EDIT ) - { - oldval = db.cell("select " + pSortfield + " from " + pTable + " where " + pTable + "ID = '" + id + "'"); - - if(pDirection == "down") - minmax = "max"; - - minmax = db.cell("select " + minmax + "(" + pSortfield + ") from " + pTable + pCondition); - } - return (oldval != minmax || oldval == "" || minmax == "" ) - } +/** +* removes an specific element from an array +* +* @param {Array} targetArray Array from which the element should be removed +* @param {String} elementPos index of the element which should be removed +* +* @return {Array} array containing the deleted element +*/ +ArrayUtils.removeElement = function(targetArray, elementPos) { + return targetArray.splice(elementPos, 1); } + /** - * Class containing utility functions for use with base64 and binaries - * @class + * concats arrays column by column; + * see example for more details + * + * @param {Array} array1 you have to pass at least 2 Arrays that shall be concated but you can pass as many as you want + * @param {Array} arrayN you have to pass at least 2 Arrays that shall be concated but you can pass as many as you want + * + * @return {Array} concatenated array + * + * @example + * var a = ["a", "b", "c"]; + * var b = ["A", "B", "C"]; + * logging.show(JSON.stringify(concatArrayColumns(a, b))); + * //[["a","A"],["b","B"],["c","C"]] + * + * logging.show(JSON.stringify(a.concat(b))); + * //["a","b","c","A","B","C"] */ -function BinaryUtils() -{ - var that = this; - /** - * decodes 64Based String - * - * @param {String} input req the string - * - * @return {String} decoded String - * - * - */ - this.decode64 = function(input) - { - var charset = new Configuration().getOption("Base64Charset"); +ArrayUtils.concatColumns = function(array1, arrayN) { + var res, i, ii, l, ll, inpArr; + res = []; - if ( charset == "" ) - charset = "ISO-8859-15"; - return util.decodeBase64String(input, charset); - } - /** - * returns the original size of a B64 coded String in bytes - * - * @param {String} pBase64str req base64 encoded string - * - * @return {Number} size of the string in bytes - * - * - */ - this.getBase64lengthBytes = function(pBase64str) - { - var res = pBase64str.length; - res = eMath.divDec(res, 4); - res = Math.ceil(res); - res = eMath.mulInt(3, res); - - return +res;//cast und return - } - - /** - * sanitizes the filelist gotten from Drag&Drop and returns it. - * - * @param {String[]} pFilelist req Die Filelist als Stringarray. - * - * - */ - this.sanitizeFilelist = function(pFilelist) - { - var sanitized = []; - - if(pFilelist != undefined && pFilelist.length > 0) - { - for(let i = 0; i < pFilelist.length; i++) - { - if(swing.doClientIntermediate(swing.CLIENTCMD_FILEIO_ISDIRECTORY, [pFilelist[i]]) == "false") - sanitized.push(pFilelist[i]); - } + for (i = 0, l = arguments.length; i < l; i++) {//this function can handle an "unlimited" amount of functionparams + inpArr = arguments[i]; + for (ii = 0, ll = inpArr.length; ii < ll; ii++) { + if (res[ii] == undefined) + res[ii] = []; + + res[ii] = res[ii].concat(inpArr[ii]); } - - return sanitized; } - /** - * reads the folders in a filelist and returns their contained files - * - * @param {String[]} pFilelist req Die Filelist als Stringarray. - * - * - */ - this.resolveSubfolders = function(pFilelist) - { - //read systemconfig for D&D - var cfg = new Configuration(); - var filesize = (cfg.getOption("dnd.maxFileSize") *1000 *1000 ); - var fileamount = 30; - var files = []; - var folders = []; + return res; +} - if(pFilelist != undefined && pFilelist.length > 0) - { - //separate files and folders - for (let i = 0; i < pFilelist.length; i++) - { - if(swing.doClientIntermediate(swing.CLIENTCMD_FILEIO_ISDIRECTORY, [pFilelist[i]]) == "false") - files.push(pFilelist[i]); - else - folders.push(pFilelist[i]); - } - //resovle Files in Folders - if(folders.length > 0) - { - for (let j = 0; j < folders.length; j++) - { - var resolvedFiles = swing.doClientIntermediate(swing.CLIENTCMD_FILEIO_LISTFILES, [folders[j]]); - if(resolvedFiles.length > 0) - { - for (let k = 0; k < resolvedFiles.length; k++) resolvedFiles[k] = folders[j] + "\\" + resolvedFiles[k]; - files = files.concat(resolvedFiles); - } +/** + * returns if an element is in an array; + * this is needed because there is currently no support for Array.prototype.includes() and we cannot easily use a polyfill and extend the Array.prototype + * + * @param {Array} targetArray array where the element should be searched in + * @param {any primitive type} element the element which should be looked for + * @param {Boolean} [ignoreStringCase=false] if you've got a string array y + * + * @return {Boolean} true if it has the element + */ +ArrayUtils.hasElement = function(targetArray, element, ignoreStringCase) { + var i, l; + if (ignoreStringCase)//do only once to save ressources and not for every array element + element = element.toString().toLowerCase(); + for (i = 0, l = targetArray.length; i < l; i++) { + if (ignoreStringCase) { + if (targetArray[i].toString().toLowerCase() == element) + return true; + } + else { + if (targetArray[i] == element) + return true; + } + } + return false; +} + +/** + * Class containing utility functions for use with JSON + * @class + * @static + */ +function JSONUtils() { +} + +/** + * A custom JSON.stringify() to + * - keep the functions as string + * - stringify JavaArrays + * - stringify undefined as undefined and not as null + * + * @param {Object} obj the object to stringify + * + * @return {String} the stringified object as string representation + */ +JSONUtils.customStringify = function(obj) { + //stringify part from JSON polyfill: https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/JSON + var toString = Object.prototype.toString; + var escMap = { + '"': '\\"', + '\\': '\\\\', + '\b': '\\b', + '\f': '\\f', + '\n': '\\n', + '\r': '\\r', + '\t': '\\t' + }; + var escFunc = function (m) { + return escMap[m] || '\\u' + (m.charCodeAt(0) + 0x10000).toString(16).substr(1); + }; + var escRE = /[\\"\u0000-\u001F\u2028\u2029]/g; + + var stringify = function (value){ + //because: "undefined == null" is true + if (value === undefined) { + return 'undefined'; + } else if (value == null) { + return 'null'; + } else if (typeof value === 'number') { + return isFinite(value) ? value.toString() : 'null'; + } else if (typeof value === 'boolean') { + return value.toString(); + } else if (typeof value === 'object') +{ + //check with "hasOwnProperty" because in JavaArrays using value.toJSON would cause an exception + if (toString.call(value) != '[object Map]' && value.hasOwnProperty("toJSON") && typeof value.toJSON === 'function') { + return value.toString(); + } else if (toString.call(value) === '[object Array]') { + var res = '['; + for (var i = 0; i < value.length; i++) + res += (i ? ', ' : '') + stringify(value[i]); + return res + ']'; + } else if (toString.call(value) === '[object Object]' || toString.call(value) === '[object Map]') { + var tmp = []; + for (var k in value) { + if (hasOwnProperty.call(value, k)) + tmp.push(stringify(k) + ': ' + stringify(value[k])); } + return '{' + tmp.join(', ') + '}'; + //custom addition to stringify Rhino JavaArrays + } else if (toString.call(value) === '[object JavaArray]') { + return value.toSource().toString(); } - - //recursive call, cause more subfolders may be present - if( that.hasSubfolders(files) ) - files = that.resolveSubfolders(files); } - - //filter out files, that exceed the size limit - var retfiles = []; - for (let m = 0; m < files.length; m++) + //custom addition for function transform + //JSON.stringify returns null on functions (default) + //instead return source code of funciton for callback-functions + else if (typeof value === 'function') { - if(retfiles.length == fileamount) - break; - - if(swing.doClientIntermediate(swing.CLIENTCMD_FILEIO_GETLENGTH, [files[m]]) <= filesize) - retfiles.push(files[m]); + return value.toString(); } - logging.log(retfiles) - return retfiles; + return '"' + value.toString().replace(escRE, escFunc) + '"'; } - /** - * Checks if subfolders are present in a filelist - * - * @param {String[]} pFilelist req the file list as string array - * - * - */ - this.hasSubfolder = function(pFilelist) - { - var ret = false; - if(pFilelist != undefined && pFilelist.length > 0) - { - for (let i = pFilelist.length; i >= 0; i--) - { - if(swing.doClientIntermediate(swing.CLIENTCMD_FILEIO_ISDIRECTORY, [pFilelist[i]]) == "true") - { - ret = true; - break; - } - } - } - return ret; + return stringify(obj); +} + +/** + * Class containing functions for sequential numbers + * @class + * @static + */ +function NumberSequencingUtils(){ +} + +/** + * Delivers the next unique number + * + * @param {String} pColumn database column that contains unique numbers + * @param {String} pTable database table + * @param {Number} [pStartNumber=1000] number to start numeration + * @param {String} [pCondition=no condition] SQL Where Conditon + * + * @result {String} next valid number + */ +NumberSequencingUtils.getNextUniqueNumber = function(pColumn, pTable, pStartNumber, pCondition) { + var maxNum = NumberSequencingUtils.getMaxUniqueNumber(pColumn, pTable, pCondition); + if(maxNum == "0") { + if(pStartNumber == undefined) pStartNumber = 1000; + return pStartNumber; } + return eMath.addInt(maxNum, "1");//increment currently highest number +} +/** + * Checks if the passed number is valid (has to be unique) + * + * @param {String} pNumber number to check + * @param {String} pColumn req database column that contains unique numbers + * @param {String} pTable req database table + * @param {String} pCondition opt SQL Where Conditon + * + * @result {boolean} passed number is valid + */ +NumberSequencingUtils.validateUniqueNumber = function(pNumber, pColumn, pTable, pCondition) { + var maxNum = NumberSequencingUtils.getMaxUniqueNumber(pColumn, pTable, pCondition); + return Number(pNumber) > Number(maxNum); +} + +/** + * Delivers the hightest number currently stored in database + * + * @param {String} pColumn req database column that contains unique numbers + * @param {String} pTable req database table + * @param {String} pCondition opt SQL Where Conditon + * + * @result {String} hightest number + */ +NumberSequencingUtils.getMaxUniqueNumber = function(pColumn, pTable, pCondition) { + var condition = ""; + if(pCondition != undefined) + condition += " where " + pCondition; + var maxNum = db.cell("select max(" + pColumn + ") from " + pTable + condition); + return maxNum == "" ? "0" : maxNum; } + /** * Class containing miscellaneous utiltiy function extending JDito * @class + * @deprecated + * @todo: remove */ function JDitoUtils() { - var that = this; - /** - * A wrapper for process.executeScript - * - * @param {String} pIdentifier req The identifier for the script (for better error logging) - * @param {String} pScript req The script to execute - * @param {Object} pLocalVariables opt The local variables which can be accessed within the script - * @param {Array} pImports opt To add additional imports which will be prepended to the script - * @param {Boolean} pWrapIntoResult opt To wrap pScript into 'result.string("***")' - * @param {String} pAlias opt The alias to use within the script - * - * @return {Object} The result of the script - * - */ - this.evalScript = function(pIdentifier, pScript, pLocalVariables, pImports, pWrapIntoResult, pAlias) - { - var defaultImports = ["system.result", "system.vars"]; - var importsToPrepend = []; - - // add pImports to to defaultImports - if (pImports != undefined) defaultImports = defaultImports.concat(pImports); - - // go trough all defaultImports an check if they already exists - for (var i = 0; i < defaultImports.length; i++ ) - { - var cImport = "import(\"" + defaultImports[i] + "\");"; - - if (pScript.indexOf(cImport) == -1) - { - // If the current default import was not found within the script it will be prepended - importsToPrepend.push(cImport); - } - } - - // join the imports to prepend and place them in front of a new variable - var script2Execute = importsToPrepend.join("\r\n") + (importsToPrepend.length > 0 ? "\r\n" : ""); - - // Check if the script needs to be wrapped into result.string() (e.g. (function(){return "result";})()) - if (pWrapIntoResult) - script2Execute += "\r\nresult.string(" + pScript + ");\r\n"; // Wrap into 'result.string("***")' - else - script2Execute += pScript; // Just add the script after the prepended imports - - return process.executeScript("[evalScript]" + pIdentifier, script2Execute, (pLocalVariables != undefined && pLocalVariables != null ? pLocalVariables : {}), (pAlias != undefined && pAlias != null ? pAlias : null)); - } - /** * Returns a list of aliasNames with a given type * ignores errors when an alias is not confgiured (empty) @@ -863,617 +451,36 @@ function JDitoUtils() return dbAliases; } - - /** - * returns a sorted and translated list of countries ; - * DE, AT and CH are at the beginning of the list, if pData is undefined - * - * @param {String} pLanguage opt Sprache - * @param {String[][]} pData opt if the data should not be gathered automatically - * it can be given to this parameter. - * with this, the function can be used to translate country names - * - * @return {String[][]} consisting of ISO2 and Name - * - * - */ - this.getCountries = function(pLanguage, pData) - { - var data, temp; - if(pData == undefined) - { - data = [ ["DE", "Deutschland"], ["AT", "Österreich"], ["CH", "Schweiz"] ]; - temp = db.table("select COUNTRYINFO.ISO2, COUNTRYINFO.NAME_DE " - + "from COUNTRYINFO " - + "where COUNTRYINFO.ISO2 not in ('DE', 'AT', 'CH')"); - } - else - { - data = []; - temp = pData; - } - - for(var i = 0; i < data.length; i++) - data[i][1] = translateStr(data[i][1], pLanguage); - for(let i = 0; i < temp.length; i++) - temp[i][1] = translateStr(temp[i][1], pLanguage).trim(); - - //Sortierung nach der Übersetzung - array_mDimSort(temp, 1, true, false); - - return data.concat(temp); - } - - /** - * Delivers the next unique number - * - * @param {String} pColumn req database column that contains unique numbers - * @param {String} pTable req database table - * @param {Number} pStartNumber opt number to start numeration - * @param {String} pCondition opt SQL Where Conditon - * - * @result {String} next valid number - */ - this.getNextUniqueNumber = function(pColumn, pTable, pStartNumber, pCondition){ - var maxNum = that.getMaxUniqueNumber(pColumn, pTable, pCondition); - - if(maxNum == "0") - { - if(pStartNumber == undefined) pStartNumber = 1000; - return pStartNumber; - } - - return eMath.addInt(maxNum, "1");//increment currently highest number - } - - /** - * Checks if the passed number is valid (has to be unique) - * - * @param {String} pNumber number to check - * @param {String} pColumn req database column that contains unique numbers - * @param {String} pTable req database table - * @param {String} pCondition opt SQL Where Conditon - * - * @result {boolean} passed number is valid - */ - this.validateUniqueNumber = function(pNumber, pColumn, pTable, pCondition){ - var maxNum = that.getMaxUniqueNumber(pColumn, pTable, pCondition); - - return Number(pNumber) > Number(maxNum); - } - - /** - * Delivers the hightest number currently stored in database - * - * @param {String} pColumn req database column that contains unique numbers - * @param {String} pTable req database table - * @param {String} pCondition opt SQL Where Conditon - * - * @result {String} hightest number - */ - this.getMaxUniqueNumber = function(pColumn, pTable, pCondition){ - - var condition = ""; - if(pCondition != undefined) - condition += " where " + pCondition; - - var maxNum = db.cell("select max(" + pColumn + ") from " + pTable + condition); - - return maxNum == "" ? "0" : maxNum; - } } -/** - * Class containing utility functions for use with JSON - * @class - */ -function JSONUtils() -{ - /** - * A custom JSON.stringify() to - * - keep the functions as string - * - stringify JavaArrays - * - stringify undefined as undefined and not as null - * - * @param {Object} pObj req The object to stringify - * - * @return {String} The stringified string - * - * - */ - this.stringifyJSON = function(pObj) - { - //stringify part from JSON polyfill: https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/JSON - var toString = Object.prototype.toString; - var escMap = {'"': '\\"', '\\': '\\\\', '\b': '\\b', '\f': '\\f', '\n': '\\n', '\r': '\\r', '\t': '\\t'}; - var escFunc = function (m) { return escMap[m] || '\\u' + (m.charCodeAt(0) + 0x10000).toString(16).substr(1); }; - var escRE = /[\\"\u0000-\u001F\u2028\u2029]/g; - - var stringify = function (value){ - //because: "undefined == null" is true - if (value === undefined) { - return 'undefined'; - } else if (value == null) { - return 'null'; - } else if (typeof value === 'number') { - return isFinite(value) ? value.toString() : 'null'; - } else if (typeof value === 'boolean') { - return value.toString(); - } else if (typeof value === 'object') - { - //check with "hasOwnProperty" because in JavaArrays using value.toJSON would cause an exception - if (toString.call(value) != '[object Map]' && value.hasOwnProperty("toJSON") && typeof value.toJSON === 'function') { - return value.toString(); - } else if (toString.call(value) === '[object Array]') { - var res = '['; - for (var i = 0; i < value.length; i++) - res += (i ? ', ' : '') + stringify(value[i]); - return res + ']'; - } else if (toString.call(value) === '[object Object]' || toString.call(value) === '[object Map]') { - var tmp = []; - for (var k in value) { - if (hasOwnProperty.call(value, k)) - tmp.push(stringify(k) + ': ' + stringify(value[k])); - } - return '{' + tmp.join(', ') + '}'; - //custom addition to stringify Rhino JavaArrays - } else if (toString.call(value) === '[object JavaArray]') { - return value.toSource().toString(); - } - } - //custom addition for function transform - //JSON.stringify returns null on functions (default) - //instead return source code of funciton for callback-functions - else if (typeof value === 'function') - { - return value.toString(); - } - - return '"' + value.toString().replace(escRE, escFunc) + '"'; - } - - return stringify(pObj); - } - /** - * checks if pStr is JSON - * - * @param {pStr} - * - * @return false or parsed JSON - * - * - */ - this.isJSON = function(pStr) - { - if (typeof(pStr) != "string") - return false; - - if (pStr[0] != "{") - return false; - - if (pStr[pStr.length-1] != "}") - return false; - - try - { - var parse = JSON.parse(pStr); - return parse; - } - catch (ex) - { - return false; - } - return parse; - } -} /** - * Class containing utility functions for copying modules + * Class containing String utility functions * @class + * @deprecated + * @todo: remove this funciton */ -function CopyModuleUtils() +function StringUtils(){} { - var that = this; /** - * opens the new created modules in neonClient - * - * @param {String} pNeonContext req Name of the neon context that should be opened - * @param {Object} pModulesMapping req ModulesMapping object created by method copyModule - * - * @example var ModulesMapping = CMUtils.copyModule(InputMapping); - * - * CMUtils.openNewModules("Offer_context", ModulesMapping); - */ - this.openNewModules = function(pNeonContext, pModulesMapping) - { - if(pModulesMapping != undefined) - { - var rootModule = Object.keys(pModulesMapping)[0]; - if(pModulesMapping[rootModule].DataRows != undefined) - { - var newids = []; - - for(var row in pModulesMapping[rootModule].DataRows) - { - newids.push(pModulesMapping[rootModule].DataRows[row].newPrimaryKey); - } - - if(newids.length > 0) - neon.openContext(pNeonContext, newids, neon.OPERATINGSTATE_VIEW, null); - } - } - } - - /** - * Creates a copy of a specified module together with specified subordinated modules. <br> - * The structure of the input mapping object is the following: <br> - * pInputMapping { - * (only one rootModule allowed) - * $rootModule$: { - * condition: "sqlWhereCondition" - * , ValueMapping: {$colName$: "value"} - * , destinationModuleName: "destinationModuleName" - * , DestinationColumnMapping: { $sourceColName$ : "destinationColName" } - * , SubModules: { - * $Module$:{ - * condition: "sqlWhereCondition" - * , ValueMapping: {$colName$: "value"} - * , destinationModuleName: "destinationModuleName" - * , DestinationColumnMapping: { $sourceColName$ : "destinationColName" } - * , SubModules: {...} - * } - * } - * } - * } - * - * @param {Object} pInputMapping - * - * @example var CMUtils = new CopyModuleUtils(); + * uses the right translate method, depending on the parameters * - * var InputMapping = { + * @param {String} pText string to be translated + * @param {String} pLocale locale for translating * - * "OFFER": { - * condition: "OFFERID = '" + vars.get("$field.OFFERID") + "'" - * ,SubModules:{ - * "OFFERITEM": { - * condition: "OFFER_ID = '" + vars.get("$field.OFFERID") + "' order by ITEMSORT" - * } - * } - * } - * } + * @return {String} * - * CMUtils.copyModule(InputMapping); + * */ - this.copyModule = function(pInputMapping) + this.translateStr = function( pText, pLocale ) { - var AliasDefinitionStructure = project.getAliasDefinitionStructure("Data_alias", null); - var ModulesMapping = {}; - var statements = []; - - buildMapping( pInputMapping ); - buildStatements( ModulesMapping ); - - if(statements.length > 0) - db.inserts( statements ); - - return ModulesMapping; - - /** - * Builds a mapping Object for the copyModule Method. <br> - * The structure of the Object is the following: <br> - * ModulesMapping = { - * - * $rootModule$: { ($$ marks an object property) - * - * (ModuleMapping) - * name: "moduleName" - * , destinationModuleName: "destinationModuleName" - * , DestinationColumnMapping: { $sourceColName$ : "destinationColName" } - * , ValueMapping: {$colName$: "value"} - * , dataRows:{ - * (ModuleRowMapping) - * $rowid$: { - * name: "moduleName" - * , oldPrimaryKey: "oldPrimaryKeyValue" - * , newPrimaryKey: "newPrimaryKeyValue" - * , ColumnMapping: { $colName$: { newValue: "newValue", oldValue: "oldValue", destinationColumn: "destinationColumn" } } - * , ModuleMapping: object reference to ModuleMapping object that contains ModuleRowMapping objects - * , ParentModuleMapping: object reference to supervised ModuleMapping object (at this point "null" because there is no supervised object) - * } - * } - * , SubModules: { - * (ModuleMapping) - * $moduleName$: { - * name: "moduleName" - * , destinationModuleName: "destinationModuleName" - * , DestinationColumnMapping: { $sourceColName$ : "destinationColName" } - * , ValueMapping: {$colName$: "value"} - * , dataRows:{ - * (ModuleRowMapping) - * $rowid$: { - * name: "moduleName" - * , oldPrimaryKey: "oldPrimaryKeyValue" - * , newPrimaryKey: "newPrimaryKeyValue" - * , ColumnMapping: { $colName$: { newValue: "newValue", oldValue: "oldValue", destinationColumn: "destinationColumn" } } - * , ModuleMapping: - * , ParentModuleMapping: object reference to supervised ModuleMapping object (at this point "null" because there is no supervised object) - * } - * } - * , SubModules: {...} - * } - * } - * - * } - * - *} - * - * @param {Object} pInputMapping InputMapping - */ - function buildMapping(pInputMapping) - { - //root mapping - var rootModule = Object.keys(pInputMapping)[0]; - var ModuleMapping = _ModuleMapping(rootModule, pInputMapping[rootModule]); - var ModuleData = _getModuleData(rootModule, pInputMapping[rootModule].condition); - - for(var row in ModuleData) - { - var ModuleRowMapping = _ModuleRowMapping(ModuleMapping, null, ModuleData[row]); - - ModuleMapping.DataRows[ModuleRowMapping.oldPrimaryKey] = ModuleRowMapping; - } - - ModulesMapping[rootModule] = ModuleMapping; - - //recursive subordinated modules mapping - _buildSubordinatedMapping(pInputMapping, ModuleMapping); - - - //delivers stored data for module in Database with condition - function _getModuleData(pModule, pCondition) - { - if(pModule == undefined) return {}; - - var ModuleColumnsStructure = AliasDefinitionStructure.tables[pModule].columns; - var cols = Object.keys(ModuleColumnsStructure); - - var condition = "1=1"; - if(pCondition != undefined) - condition = pCondition; - - var dbData = db.table("select " + cols.join(", ") + " from " + pModule + " where " + condition); - - //map 2d-Array to Object { $rowNumber$: { $columnName$: { value: "valueInDB" } } } - var DataObj = {}; - for(var row = 0; row < dbData.length; row++) - { - DataObj[row] = {}; - for(var col = 0; col < dbData[row].length; col++) - { - DataObj[row][cols[col]] = { - value: dbData[row][col] - }; - } - } - - return DataObj; - } - - - //recursive: ModuleMapping and ModuleRowMapping for subordinated modules - function _buildSubordinatedMapping(pInputMapping, pParentModuleMapping) - { - var SubModules = pInputMapping[pParentModuleMapping.name].SubModules; - if(SubModules == undefined) - return; - - for(var subModuleName in SubModules) - { - var ModuleMapping = _ModuleMapping(subModuleName, SubModules[subModuleName]); - ModuleData = _getModuleData(subModuleName, SubModules[subModuleName].condition); - for(var row in ModuleData) - { - ModuleRowMapping = _ModuleRowMapping(ModuleMapping, pParentModuleMapping, ModuleData[row]); - ModuleMapping.DataRows[ModuleRowMapping.oldPrimaryKey] = ModuleRowMapping; - } - - ModulesMapping[pParentModuleMapping.name].SubModules[subModuleName] = ModuleMapping; - - _buildSubordinatedMapping(SubModules, ModuleMapping); - } - } - - function _ModuleMapping( pModuleName, pInputModuleMapping ) - { - return { - name: pModuleName, - destinationModuleName: pInputModuleMapping.destinationModuleName == undefined ? pModuleName : pInputModuleMapping.destinationModuleName, - DestinationColumnMapping: pInputModuleMapping.DestinationColumnMapping == undefined ? {} : pInputModuleMapping.DestinationColumnMapping, - ValueMapping: pInputModuleMapping.ValueMapping == undefined ? {} : pInputModuleMapping.ValueMapping, - DataRows:{}, - SubModules: {} - }; - } - - function _ModuleRowMapping(pModuleMapping, pParentModuleMapping, pDataRow) - { - var ModuleRowMapping = { - name: pModuleMapping.name, - oldPrimaryKey: null, - newPrimaryKey: null, - ColumnMapping: {}, - ModuleMapping: pModuleMapping, - ParentModuleMapping: pParentModuleMapping - }; - - var ModuleColumnsStructure = AliasDefinitionStructure.tables[ModuleRowMapping.name].columns; - - //build ColumnMapping - for(var col in ModuleColumnsStructure) - { - //set defined columns from InputMapping -> if not defined, use the same column - var destinationColumn = ModuleRowMapping.ModuleMapping.DestinationColumnMapping[col]; - if(destinationColumn == undefined) - destinationColumn = col; - - //set defined values from InputMapping -> if not defined, use the value from DB - var oldValue = pDataRow[col].value; - var newValue = newValue = ModuleRowMapping.ModuleMapping.ValueMapping[col]; - if(newValue == undefined) - newValue = oldValue; - - //set new primary key - if(ModuleColumnsStructure[col].primaryKey) - { - ModuleRowMapping.oldPrimaryKey = ModuleRowMapping.ColumnMapping[col] = oldValue; - newValue = util.getNewUUID(); - ModuleRowMapping.newPrimaryKey = newValue; - } - - ModuleRowMapping.ColumnMapping[col] = _columnMapping(newValue, oldValue, destinationColumn); - } - - switch(ModuleRowMapping.name) - { - case "OFFER": - { - //andere Values setzen - var dtUtils = new DateUtils(); - ModuleRowMapping.ColumnMapping["OFFERDATE"].newValue = dtUtils.getTodayUTC(); - } - break; - case "OFFERITEM": - { - //OFFER_ID mappen - if(ModuleRowMapping.ParentModuleMapping.name == "OFFER") - { - ModuleRowMapping.ColumnMapping["OFFER_ID"].newValue = ModulesMapping[ModuleRowMapping.ParentModuleMapping.name].DataRows[ModuleRowMapping.ColumnMapping["OFFER_ID"].oldValue].newPrimaryKey; - } - //ASSIGNEDTO mappen - if(ModuleRowMapping.ColumnMapping["ASSIGNEDTO"].oldValue != "") - { - if(ModuleRowMapping.ParentModuleMapping == null) - { - ModuleRowMapping.ColumnMapping["ASSIGNEDTO"].newValue = ModulesMapping["OFFERITEM"].DataRows[ModuleRowMapping.ColumnMapping["ASSIGNEDTO"].oldValue].newPrimaryKey; - } - else - { - ModuleRowMapping.ColumnMapping["ASSIGNEDTO"].newValue = ModuleRowMapping.ModuleMapping.DataRows[ModuleRowMapping.ColumnMapping["ASSIGNEDTO"].oldValue].newPrimaryKey; - } - } - } - break; - case "OFFERLINK": - { - - } - default: - { - - } - - - } - - //Spezialbehandlung USER_NEW DATENEW.... - if(ModuleRowMapping.ColumnMapping["DATE_NEW"] != undefined) ModuleRowMapping.ColumnMapping["DATE_NEW"].newValue = datetime.date(); - if(ModuleRowMapping.ColumnMapping["USER_NEW"] != undefined) ModuleRowMapping.ColumnMapping["USER_NEW"].newValue = vars.get("$sys.user"); - if(ModuleRowMapping.ColumnMapping["DATE_EDIT"] != undefined) ModuleRowMapping.ColumnMapping["DATE_EDIT"].newValue = ""; - if(ModuleRowMapping.ColumnMapping["USER_EDIT"] != undefined) ModuleRowMapping.ColumnMapping["USER_EDIT"].newValue = ""; - - return ModuleRowMapping; - } - - - - - function _columnMapping(pNewValue, pOldValue, pDestinationColumn) - { - return { - newValue: pNewValue, - oldValue: pOldValue, - destinationColumn: pDestinationColumn - }; - } - - } - - /** - * Builds the insert statements for passed ModulesMapping - * - * @param {Object} pModulesMapping ModulesMapping from buildMapping() - */ - function buildStatements(pModulesMapping) - { - var rootModule = Object.keys(pModulesMapping)[0]; - - for(var row in pModulesMapping[rootModule].DataRows) - { - //buildInsertStatement - statements.push(_statement(pModulesMapping[rootModule].DataRows[row])); - } - - _subordinatedStatements(pModulesMapping[rootModule]); - - function _subordinatedStatements(pMapping) - { - if(pMapping.SubModules == undefined) - return; - - for(var subModule in pMapping.SubModules) - { - - for(var row in pMapping.SubModules[subModule].DataRows) - { - statements.push(_statement(pMapping.SubModules[subModule].DataRows[row])); - } - - _subordinatedStatements(pMapping.SubModules[subModule]); - } - - } - - function _statement(pRowMapping) - { - var cols = []; - var vals = []; - var destTable = pRowMapping.ModuleMapping.destinationModuleName; - var colMapping = pRowMapping.ColumnMapping; - - for(var col in colMapping) - { - cols.push(colMapping[col].destinationColumn); - vals.push(colMapping[col].newValue.toString()); - } - - var colTypes = db.getColumnTypes(destTable, cols) - - return [destTable, cols, colTypes, vals]; - } - } + if ( pLocale == undefined ) + return translate.text(pText); + else + return translate.text(pText, pLocale) } -} - -/** -* provides somehow static methods for special handling in JDito-Processes -* do not create an instance of this -* @class -*/ -ProcessHandlingUtil = { - /** - * In onValidation-Process a local variable called "$local.value" is made available from kernel. - * It contains the entered value - the field contains the value only after successfull validation (vars.get("$field.Fieldname")). - * The onValidation-Process is running again before saving the entity, at this point there's "$local.value" varialbe no longer available, - * but the entered value now is the present one because the field has already been validated before. - * Otherwise a "variable not found" error would occur. - * - * @param {String} pFieldValue req value of the field onValidation-Process is executed ( e.g. vars.get("$field.Fieldname") ) - * @return {String} Field value for onValidation-Process - */ - getOnValidationValue: function(pFieldValue) - { - return vars.exists("$local.value") ? vars.get("$local.value") : pFieldValue; - } + } +