Skip to main content

Financial Dimensions and Lookups to Excel Templates


Before we begin, let's confirm that we don't have financial dimensions enabled through OData yet. The best way to check is by confirming if the fields are available in Excel. You can do this with the following steps:Open your Excel template from Dynamics by clicking the Design button for a dimension-enabled template, such as general journal or budget register.

In the Data Connector edit the template for lines (journal lines or budget register entry lines)
As you can see here, I don't have a field for Account.Department, which is a dimension I use in my company, Contoso.



Secondly, you can have an administrator run the below query and confirm if the columns exist or not in the DimensionCombinationEntity table. This table is used in the Excel add-in so missing dimension columns also indicate the dimensions aren't available for use in Excel.

Select top 100 * from DIMENSIONCOMBINATIONENTITY



Now that we know the dimensions aren't available we'll proceed on to add our necessary dimensions.
Enable dimensions through Odata

We will create a new model. You may already have one or you may need to create one if you don't have one with the correct dependencies. ApplicationPlatform
  • Directory
  • Dimensions
  • Source Documentation



With our model built, we will click Addins > Add financial dimensions for OData. In this screen, you need to enter the values exactly as they are defined in the financial dimensions for integrating applications under the General Ledger module in Dynamics. You also need to select your model to use for the customization. In my case, I am adding dimensions for BusinessUnit, Department, and CostCenter. The wizard will generate a project in Visual Studio that we need to build, and then synchronize to the database. Once synchronization is complete, we can test the results.



If we open our journal entry in Excel again and check the available fields, we now see available fields for Account.BusinessUnit, Account.Department, and Account.CostCenter in addition to a few other fields for the corresponding offset account dimensions.



We need to move the fields to be selected and included in the template. If we click within a table row for the new fields we can now enter a value or read back the value from anything added in the client.



If we check the DimensionCombinationEntity table again we'll see results that correlate with what we see in Excel. Now we have additional columns for these three dimensions.


Enable dimension value lookups through Odata

We now have the fields available for us to enter data but there are no lookups for us to figure out what the dimension values are or what they represent. We'll continue editing our previous Visual Studio project and enable lookups as the next step.

In the same project, right-click to open DimensionCombinationEntity in the right-hand pane. Create a relation for each dimension, following the steps below.

In the Properties pane, set the following properties.

Validate - No


Cardinality - ZeroMore


Name - Enter the name of the financial dimension, such as Department, BusinessUnit, or CostCenter.


Related Data Entity - Select the entity for the financial dimension that you entered in the Name field. For my dimensions, I needed DimAttributeOMDepartmentEntity, DimAttributeOMBusinessUnitEntity, and DimAttributeOMCostCenterEntity


Related Data Entity Cardinality - ZeroOne


Related Data Entity Role - Enter a unique name, such as "Dimension Department, BusinessUnit, or CostCenter Lookup".


Relationship Type - Association


Role - Enter a unique name, such as Dimension Department, BusinessUnit, or CostCenter.

Then add another relation to represent the second part of the link.

Right-click the Financial dimension name under Relations.


Select New, and then click Normal.


In the Properties pane, choose the name of the Financial dimension in the Field.
In the Related field, type Value.

Once we've completed the steps for all three dimensions you can build the project and then synchronize.

I made two mistakes when recording this. The first is that I failed initially to add a dependency in my model to Source Document Framework. Because of this, I didn't see the table in the lookup and receive an error Table DimAttributeOM...Entity does not exist.

The other error I saw was Property cannot be empty, which made me re-review my steps and found I left Related Field blank instead of Value on the second part of the relationship.

Now that this is synchronized without error we can navigate back to the client. We will update the Excel template and see our lookup change. I didn't make the change here, but we can save our updated template and upload it to Document templates within Organization Administration to make our new dimension column addition permanent in Excel.


Now to add custom financial dimension lookups in filtered columns we can create normal relations as above but use below eventhandler code as well to apply filter on custom dimensions:


[PostHandlerFor(classStr(DimensionSegmentGenericLookupGenerator), staticMethodStr(DimensionSegmentGenericLookupGenerator, generateCustomLookupForDimensionSegment))]
public static void Post_generateCustomLookupForDimensionSegment(XppPrePostArgs args)
{
OfficeAppCustomLookupRelationResult customRelationLookup;
ExportToExcelFilterTreeBuilder builder;
Microsoft.Dynamics.Platform.Integration.Office.FilterBinaryNode filter;
FieldName dimensionSegment,
fieldName,
relatedExternalFieldName;

SysDictTable table;
TableId tableId,
relatedDimensionAttributeEntityId;
SysDictField field;
FieldId fieldId,
relatedFieldId,
relatedExternalFieldId,
secLineRelatedExternalFieldId;
SysDictRelation dimensionSegmentRelation;
TableName relatedDimensionAttributeEntityName;
Set tableRelations;
SetEnumerator enumerator;
SysDictRelation currentRelation;
str publicEntityName;

customRelationLookup = args.getReturnValue();


if(!customRelationLookup)
{
dimensionSegment = args.getArg(identifierStr(_dimensionSegment));

table = new SysDictTable(tableNum(DimensionCombinationEntity));
tableId = table.id();

field = new SysDictField(tableId, table.fieldName2Id(dimensionSegment));
fieldId = field.id();


tableRelations = table.relations();
enumerator = tableRelations.getEnumerator();


while (enumerator.moveNext())
{
currentRelation = enumerator.current();


relatedFieldId = currentRelation.lineTableValue(1);

if (currentRelation.lines() > 1 && relatedFieldId == fieldId)
{
dimensionSegmentRelation = currentRelation;


secLineRelatedExternalFieldId = currentRelation.lineExternTableValue(2);
break;
}
}

if (dimensionSegmentRelation && secLineRelatedExternalFieldId)
{
relatedDimensionAttributeEntityId = dimensionSegmentRelation.externTable();
relatedExternalFieldId = dimensionSegmentRelation.lineExternTableValue(1);


relatedDimensionAttributeEntityName = tableId2Name(relatedDimensionAttributeEntityId);
relatedExternalFieldName = fieldId2Name(relatedDimensionAttributeEntityId, relatedExternalFieldId);


if (relatedDimensionAttributeEntityName && relatedExternalFieldName)
{
publicEntityName = ExportToExcelMetadataCache::getEntity(relatedDimensionAttributeEntityName).PublicEntityName;


customRelationLookup = new OfficeAppCustomLookupRelationResult();
customRelationLookup.addDeterminationField(dimensionSegment);
customRelationLookup.entityName(publicEntityName);
customRelationLookup.fieldName(relatedExternalFieldName);

builder = new ExportToExcelFilterTreeBuilder(relatedDimensionAttributeEntityName);
filter = builder.areEqual(fieldId2Name(relatedDimensionAttributeEntityId, secLineRelatedExternalFieldId), dimensionSegment);

customRelationLookup.filter(filter.ToString());

args.setReturnValue(customRelationLookup);
}
}
}
}
HEY HEY HEY!!! HACK OF THE DAY!!
If more than one financial dimensions used same entity for relations always add ROLE - Relation properties 
  • Role - Enter a unique name, such as Dimension Department, BusinessUnit, or CostCenter.


Comments

Popular posts from this blog

Edit Method on Form

Edit Method D365 for a form Data Source 1. To create an edit method first create a controller class. with following properties  public static edit MainAccountNum LedgerJournalTransLedger(LedgerJournalTrans _ledgerjournal, boolean _set, MainAccountNum _id) { MainAccountNum accountId = _id; MainAccount mainAccount = MainAccount::findByMainAccountId(_id); if(_set) { if(_ledgerjournal.AccountType== LedgerJournalACType::Ledger) { mainAccount = MainAccount::findByMainAccountId(accountId); if(_ledgerjournal.LedgerDimension) { DimensionDefault defaultDim = LedgerDimensionFacade::getDefaultDimensionFromLedgerDimension(_ledgerjournal.LedgerDimension); _ledgerjournal.LedgerDimension = LedgerDimensionDefaultingEngine::getLedgerDimensionFromAccountAndDim(mainAccount.RecId, DimensionHierarchy::getAccountStructure(mainAccount.RecId), defaultDim); } else { _ledgerjournal.LedgerDimension = LedgerDimensionDefaultingEngine::getLedgerDimensionFromAccountAndDim(mainAccount.RecId, DimensionHierarchy::ge...

Lookup and Mandatory fields on Form using Form event Handlers D365

Lookup and Mandatory fields on Form using Form event Handlers D365 [FormControlEventHandler(formControlStr(VendBankAccounts, VendBankAccount_SLD_BankCode), FormControlEventType::Lookup)] public static void VendBankAccount_SLD_BankCode_OnLookup(FormControl sender, FormControlEventArgs e)     {         Query query = new Query();         QueryBuildDataSource queryBuildDataSource;         FormControlCancelableSuperEventArgs event = e as FormControlCancelableSuperEventArgs;         QueryBuildRange queryBuildRange;         SysTableLookup sysTableLookup = SysTableLookup::newParameters(tableNum(SLD_BankCodes), sender);         sysTableLookup.addSelectionField(fieldNum(SLD_BankCodes, BankCodes));         sysTableLookup.addLookupField(fieldNum(SLD_BankCodes, BankCodes));         sysTableLookup.addLookupField(fieldNum(SLD_BankCode...

Field Level Security In Ax 2012

FIELD LEVEL SECURITY IN AX 2012 In this blog we will discuss field level security applied in  Ax 2012. 1. Field level security is added on form level as well in Ax 2012 its bit different then field level security in D365 2. This also requires privileges to be applied on form specific fields only. Step #1 :  Change datasource property of the field which need to be added in security. Step #2 : Now add these 2 fields in form --- > permissions --- > Read Step #3 : Now finally create privilege and add these fields explicitly in it.       HEY HEY HEY !!!! HACK OF THE DAY  Add Fields in Entry point privilege section of form  Explicitly.