Friday, 17 September 2021

Batch class with lookup on dialog using Sys Operation Framework in D365 FSCM

We need to create 4 classes 

  • Controller class - This class works as the entry point for the process.
  • Service class - This class does the actual job.
  • Contract class - Handles the data values.
  • UI Builder class (optional) - This class is used to do modifications on the dialog (Like - 01) To provide lookup on the field, 02) To make any field mandatory or disabled).

  • Controller class

 /// <summary>
/// Controller class to generate the ER business document for customer invoices.
/// </summary>
class SalesContractCustCommunicationController extends SysOperationServiceController
{
    /// <summary>
    /// New method for GenerateSalesOrderController class.
    /// </summary>
    protected void new()
    {
        super();
        this.parmClassName(classStr(SalesContractCustCommunicationService));
        this.parmMethodName(methodStr(SalesContractCustCommunicationService, processOperation));
     
        this.parmDialogCaption("Your dialog caption");
    }
    /// <summary>
    /// Caption for batch dialogue.
    /// </summary>
    /// <returns>
    /// caption
    /// </returns>
    public ClassDescription caption()
    {
        return "Your class description";
    }
    /// <summary>
    /// Main method for controller class.
    /// </summary>
    /// <param name = "args">
    /// Args
    /// </param>
    public static void main(Args args)
    {
        SalesContractCustCommunicationController controller = new SalesContractCustCommunicationController();
     
        controller.startOperation();
    }



  • Service class

 /// <summary>

/// Service class to generate the ER business document for customer invoices.

/// </summary>


class SalesContractCustCommunicationService extends SysOperationServiceBase

{

    /// <summary>

    /// Process the actual operation for the class.

    /// </summary>

    /// <param name = "_datacontract">

    /// SalesContractCustCommunicationDataContract object

    /// </param>

    public Void processOperation(SalesContractCustCommunicationDataContract _datacontract)

    {

        #OCCRetryCount

        try

        {

            CustInvoiceJour custInvoiceJour;

            Args args = new Args();


//Note here SalesSubContractID is a custom field.

            select custInvoiceJour

                where custInvoiceJour.SalesSubContractId == _datacontract.parmSalesSubContractId();


            

            if (custInvoiceJour.RecId)

            {

                args.parmEnumType(enumnum(PrintCopyOriginal));

                args.parmEnum(1);

                args.record(custInvoiceJour);


                new MenuFunction(menuitemOutputStr(SalesInvoiceOriginal), MenuItemType::Output).run(args);

            }

        }

       

        catch (Exception::Deadlock)

        {

            // retry on deadlock

            retry;

        }

        catch (Exception::UpdateConflict)

        {

            // try to resolve update conflict

            if (appl.ttsLevel() == 0)

            {

                if (xSession::currentRetryCount() >= #RetryNum)

                {

                    throw Exception::UpdateConflictNotRecovered;

                }

                else

                {

                    retry;

                }

            }

            else

            {

                throw Exception::UpdateConflict;

            }

        }

        catch(Exception::DuplicateKeyException)

        {

            // retry in case of an duplicate key conflict

            if (appl.ttsLevel() == 0)

            {

                if (xSession::currentRetryCount() >= #RetryNum)

                {

                    throw Exception::DuplicateKeyExceptionNotRecovered;

                }

                else

                {

                    retry;

                }

            }

            else

            {

                throw Exception::DuplicateKeyException;

            }

        }

    }





  • Data Contract class

 /// <summary>

/// Contract class to generate the ER business document for customer invoices.

/// </summary>

[DataContractAttribute,

SysOperationContractProcessingAttribute(classStr(SalesContractCustCommunicationUIBuilder))]

class SalesContractCustCommunicationDataContract

{

  // Note, here SalesSubContractId is a custom field

    SalesSubContractId  salesSubContractId;

    

    /// <summary>

    /// sub framwork parameter for Release batch job.

    /// </summary>

    /// <param name = "_salesSubContractId">

    /// Sales sub contract Id

    /// </param>

    /// <returns>

    /// Sales sub contract Id value

    /// </returns>

    [DataMemberAttribute,

    SysOperationLabelAttribute("Your Label Attribute Value"),

    SysOperationHelpTextAttribute("Your Help Text Attribute Value"),

    SysOperationDisplayOrderAttribute('1')]

    public SalesSubContractId parmSalesSubContractId(SalesSubContractId _salesSubContractId = salesSubContractId)

    {

        salesSubContractId = _salesSubContractId;

        return salesSubContractId;

    }


}



  • UI Builder class

 /// <summary>

/// UI Builder class to generate the ER business document for customer invoices.

/// </summary>


class SalesContractCustCommunicationUIBuilder extends SysOperationAutomaticUIBuilder

{

    /// <summary>

    ///    Implements a custom lookup for dimension attribute.

    /// </summary>

    /// <param name="_control">

    ///    The <c>FormStringControl</c> for which the lookup fields must be associated.

    /// </param>

    /// <remarks>

    ///    It uses the <c>SalesContractSubFCTable</c> table for lookup.

    /// </remarks>

    public void lookupSubContractId(FormStringControl _control)

    {

        Query                   query = new Query();

        QueryBuildDataSource    qbds;

        SysTableLookup          sysTableLookup;


        if (_control != null)

        {

            // NOTE: here "SalesContractSubFCTable" is a custom table

            sysTableLookup = SysTableLookup::newParameters(tablenum(SalesContractSubFCTable), _control);

            qbds = query.addDataSource(tableNum(SalesContractSubFCTable));

            SysTableLookup.addLookupfield(fieldNum(SalesContractSubFCTable, SalesSubContractId), true);

            SysTableLookup.addLookupfield(fieldNum(SalesContractSubFCTable, SalesContractId), false);

            sysTableLookup.parmQuery(query);

            sysTableLookup.performFormLookup();

        }

    }


    /// <summary>

    /// Post build method for UI Builder class

    /// </summary>

    public void postBuild()

    {

        super();


        // get datacontract

        SalesContractCustCommunicationDataContract salesContractCustCommunicationDataContract = this.dataContractObject();


        // get dialog fields

        DialogField dialogField = this.bindInfo().getDialogField(salesContractCustCommunicationDataContract, methodstr(SalesContractCustCommunicationDataContract, parmSalesSubContractId));


        dialogField.fieldControl().mandatory(true);


        // register override methods

        dialogField.registerOverrideMethod(

            methodStr(FormStringControl, lookup),

            methodStr(SalesContractCustCommunicationUIBuilder, lookupSubContractId),

            this);


    }


}


  • Now create a menu item of type Action and provide the controller class object in it.

=================================================================



Tuesday, 14 September 2021

Add Date Time Format with timestamp as HH:MM:SS:MS to FileName in D365 FSCM

 Some time we need to add the date timestamp with Mili seconds in the time format (Dynamics _ACTUALS_Sep-2020_20210909_13111510) to our output file, so below code will do it:


class Runnable1

{

    /// <summary>

    /// Runs the class with the specified arguments.

    /// </summary>

    /// <param name = "_args">The specified arguments.</param>

    public static void main(Args _args)

    {

        str     s;

        date    dateValue = DateTimeUtil::getToday(DateTimeUtil::getUserPreferredTimeZone());

        

        s = date2Str(dateValue,

                        123,

                        DateDay::None,

                        DateSeparator::Hyphen,

                        DateMonth::Short,

                        DateSeparator::Hyphen,

                        DateYear::Digits4

                      );

        

        int time = timenow();


        System.DateTime dateTime = System.DateTime::get_UtcNow();


        str utcTimeAsStr = dateTime.ToString('HHmmssff');

         // Use this format also and see the difference in timestamp value.

        //str utcTimeAsStr = dateTime.ToString('HHmmssfff');


        str customString = s + "_" + date2Str(DateTimeUtil::getSystemDate(DateTimeUtil::getUserPreferredTimeZone()),

            321, 

            2,

            DateSeparator::None,

            2,

            DateSeparator::None,

            4)+ "_"+ utcTimeAsStr;


        Info(strFmt("Format: %1", customString));

    }


}



The Output would be:

  1. Format: Sep-2021_20210914_13043935

Insert/Update or remove the default dimension value in D365 FSCM via x++

Use below method to insert/update the dimension values, just pass the parameter values to the method and it will return the updated value: p...