среда, 1 июля 2015 г.

SMART TALKS 

Dynamics AX

Design pattern fabric в Dynamics AX 2012 и варианты его использования 

Преллагаю небольшой видео блог по Design pattern fabric и Extension framework AX 2012. Введение моделей в AX 2012 позволяет более гибко использовать функциональность сторонних разработчиков, но добавляет определенные сложности с устоявшимися традициями по созданию наследников в классах, в зависимости от какого-то условия. На встрече мы обсудили какие способы решения данного вопроса существуют.

http://smart-talks.org/event/smart-talks-2/#r2


понедельник, 22 июня 2015 г.

SMART TALKS 

Dynamics AX

Change Tracking в Dynamics AX 2012


Преллагаю небольшой видео блог по Change Tracking в Dynamics AX 2012. Основные принципы работы с отслеживанием изменений, возможности, рекомендации.

http://smart-talks.org/event/smart-talks-2/


пятница, 12 июня 2015 г.

SMART TALKS 

Dynamics AX

DIXF – описание, принципы, правила 


Преллагаю небольшой видео блог по Data Import eXport Framework. Основные принципы построения модуля, основные принципы работы и использования модуля.

http://smart-talks.org/event/smart-talks-10/#r1


пятница, 5 июня 2015 г.

SMART TALKS Dynamics AX: Design pattern Посетитель видео блог

SMART TALKS 

Dynamics AX

Design pattern Посетитель


Преллагаю небольшой видео блог по Design pattern Посетитель в Dynamics AX. В примере рассмотрены базовые принципы - плюсы и минусы, варианты использования.



SMART TALKS: Dynamics AX 2012 R3 наследование таблиц видео блог

SMART TALKS

Dynamics AX 2012 R3: наследование таблиц видео блог

Преллагаю небольшой видео блог по наследованию таблиц в Dynamics AX 2012 R3. В примере рассмотрены базовые принципы - плюсы и минусы наследования.



пятница, 4 июля 2014 г.

How to transfer objects between server and client using List, Set or Map in Dynamics AX


Hi.
Some time we use such structure as a list of classes. And we can get into situation, when we need to transfer our list using a container (for example between client and server or we need to print Excel report for client ...).
In this post we will try to do this operation using very simple approach.

Starting conditions. We have client site class, and hard processing. We want to transfer hard processing to the server. Let's try do something interesting ...
CMSalesOrderDataContract with some variables

class CMSalesOrderDataContract
{
    SalesId   salesId;
    Qty       maxQty;
    Qty       averageQty;
}

we have parm methods to access these fields

public SalesId parmSalesId(SalesId _salesId = salesId)
{
    salesId = _salesId;
    return salesId;
}

public Qty parmMaxQty(Qty _maxQty = maxQty)
{
    maxQty = _maxQty;
    return maxQty;
}

public Qty parmAverageQty(Qty _averageQty = averageQty)
{
    averageQty = _averageQty;
    return averageQty;
}

and simple init method
public void initFromSalesTable(SalesTable _salesTable)
{
    SalesLine   salesLine;

    this.parmSalesId(_salesTable.SalesId);
    select maxOf(SalesQty) from salesLine
        where salesLine.SalesId == _salesTable.SalesId
        ;
    this.parmMaxQty(salesLine.SalesQty);
    select avg(SalesQty) from salesLine
        where salesLine.SalesId == _salesTable.SalesId
        ;
    this.parmAverageQty(salesLine.SalesQty);
}

and construct method
public static CMSalesOrderDataContract construct()
{
    return new CMSalesOrderDataContract();
}

Than we have 2 classes:
one collects information for us, another prints information in form;

Server class collects and saves information for us.
Declare variables
class CMCollectInformation
{
    List salesOrdersInfo;
}
init list
public void new()
{
    salesOrdersInfo = new List(Types::Class);
}
construct class on Server side.
public server static CMCollectInformation construct()

{
    return new CMCollectInformation();
}

collect information
public void collect()
{
    SalesTable                  salesTable;
    CMSalesOrderDataContract    salesOrderDataContract;
    while select salesTable
    {
        salesOrderDataContract = CMSalesOrderDataContract::construct();
        salesOrderDataContract.initFromSalesTable(salesTable);
        salesOrdersInfo.addEnd(salesOrderDataContract);
    }
}
return information packed in a container
public container getPackedData()
{
    return salesOrdersInfo.pack();
}

And second class allows us to print information on client side
class CMPrintInformation
{

}
initialize class strong on CLIENT side
public client static CMPrintInformation construct()
{
    return new CMPrintInformation();
}

public void printInfo()
{
    CMCollectInformation        collectInformation;
    ListEnumerator              le;
    CMSalesOrderDataContract    salesOrderDataContract;
    List                        soClientInfoList;

    // create and collect information n server side
    collectInformation  = CMCollectInformation::construct();
    collectInformation.collect();
    soClientInfoList = List::create(collectInformation.getPackedData());
    le = soClientInfoList.getEnumerator();

    while (le.moveNext())
    {
        salesOrderDataContract = le.current();
        info(strFmt('SalesId %1, maxQty %2, averageQty %3', salesOrderDataContract.parmSalesId(), salesOrderDataContract.parmMaxQty(), salesOrderDataContract.parmAverageQty()));
    }
}

Finally let's create a job to test our work
static void Job3(Args _args)
{
    CMPrintInformation  printInformation = CMPrintInformation::construct();
    printInformation.printInfo();
}

An error is thrown, because our DataContract class does not contain pack and unpack methods

Let's implment a few changes in our dataContract class 
Class declaration should look like

class CMSalesOrderDataContract implements SysPackable
{
    SalesId   salesId;
    Qty       maxQty;
    Qty       averageQty;
    
    #define.CurrentVersion(1)
    #localmacro.CurrentList
        salesId,
        maxQty,
        averageQty
    #endmacro
    
}

and add pack method
public container pack()
{
    return [#CurrentVersion,#CurrentList];
}

and add unpack method
public boolean unpack(container packedClass)
{
    int version     = runbase::getVersion(packedClass);
    switch (version)
    {
        case #CurrentVersion:
            [version,#CurrentList] = packedClass;
            return true;

        default :
            return false;
    }
    return false;
}

Run the job again and get a new error

We need to implement create method 

public static CMSalesOrderDataContract create(container _packedClass)
{

    CMSalesOrderDataContract    salesOrderDataContract = CMSalesOrderDataContract::construct();

    salesOrderDataContract.unpack(_packedClass);
    return salesOrderDataContract;
}

Try to run our job once again

Excellent result!

Summary:
When we want to transfer List, Set or Map containing objects of a class between server and client we have to implement pack, unpack methods and create method. Pack and unpack methods are standard. Create method must be static and should get container as an input parameter. It should create an instance of the class, unpack variables from passed container and return the instance.



пятница, 6 июня 2014 г.

Useful. Copy purchLines from one purch order to another

Copy purchLines from one purch order to another in Dynamics AX 2009, 2012.

Last time I get many question from new developer How to use ... I want to print simple but useful jobs to help newbie "Do their Things Faster With More Energy".

Standard functionality allow copy purch (sales) lines from one order to another using form \Forms\PurchCopying. If we try to see algorithm we can see construction like below.

AX2009

static void Job63(Args _args)
{
    PurchCopying            purchCopying = PurchCopying::construct(SalesPurchCopy::CopyAllLines);
    TmpFrmVirtual           tmpFrmVirtualLines;
    TmpFrmVirtual           tmpFrmVirtualHeader;
    PurchTable              purchTableFrom  = PurchTable::find('PO-0000597');
    PurchLine               purchLineFrom;
    PurchTable              purchTableTo    = PurchTable::find('PO-0000602');

    void writeTmpFrmVirtual(TmpFrmVirtual   _tmpFrmVirtual,
                            tableId         _tableId,
                            recId           _recId,
                            Num             _id,
                            LineNum         _lineNum = 0,
                            TransDate       _transDate = systemdateget(),
                            Qty             _qty = 0)

    //\Forms\PurchCopying\Methods\writeTmpFrmVirtual
    {
        _tmpFrmVirtual.TableNum     = _tableId;
        _tmpFrmVirtual.RecordNo     = _recId;
        _tmpFrmVirtual.Id           = _id;
        _tmpFrmVirtual.LineNum      = _lineNum;
        _tmpFrmVirtual.TransDate    = _transDate;
        _tmpFrmVirtual.Qty          = _qty;
        _tmpFrmVirtual.write();
    }
    ;
    // if we want to copy with header.
    //writeTmpFrmVirtual(tmpFrmVirtualHeader,
    //                   purchTableFrom.TableId,
    //                   purchTableFrom.RecId,
    //                   purchTableFrom.PurchId);
    while select purchLineFrom
        where purchLineFrom.PurchId == purchTableFrom.PurchId
    {
        tmpFrmVirtualLines.clear();
        tmpFrmVirtualLines.initValue();
        writeTmpFrmVirtual(tmpFrmVirtualLines,
                           purchLineFrom.TableId,
                           purchLineFrom.RecId,
                           purchLineFrom.PurchId,
                           purchLineFrom.LineNum,
                           systemdateget(),
                           purchLineFrom.PurchQty);
    }
    purchCopying.initParameters(purchTableTo, tmpFrmVirtualLines, tmpFrmVirtualHeader);
    purchCopying.copy(); // transaction will created inside method
}

AX2012

static void Job63(Args _args)
{
    PurchCopying            purchCopying = PurchCopying::construct(SalesPurchCopy::CopyAllLines);
    TmpFrmVirtual           tmpFrmVirtualLines;
    TmpFrmVirtual           tmpFrmVirtualHeader;
    PurchTable              purchTableFrom  = PurchTable::find('PO-000055');
    PurchLine               purchLineFrom;
    PurchTable              purchTableTo    = PurchTable::find('PO-000057', true /*not good selectforupdate within transaction, but otherwise standart 2012R2 work with error */);
    List                    tmpFrmVirtualLinesList = new List(Types::Record);
    List                    tmpFrmVirtualHeaderList = new List(Types::Record);   
    PurchCopyingPurchTableContract    contract;
    Qty                     qtyFactor         = 1;
    NoYes                   reverseSign       = NoYes::No;
    NoYes                   recalculateAmount = NoYes::No;
    NoYes                   copyMarkup        = NoYes::No;
    NoYes                   copyPrecisely     = NoYes::No;
    NoYes                   deleteLines       = NoYes::No;

    void writeTmpFrmVirtual(TmpFrmVirtual   _tmpFrmVirtual,
                            tableId         _tableId,
                            recId           _recId,
                            Num             _id,
                            LineNum         _lineNum = 0,
                            TransDate       _transDate = systemdateget(),
                            Qty             _qty = 0)

    //\Forms\PurchCopying\Methods\writeTmpFrmVirtual
    {
        _tmpFrmVirtual.TableNum     = _tableId;
        _tmpFrmVirtual.RecordNo     = _recId;
        _tmpFrmVirtual.Id           = _id;
        _tmpFrmVirtual.LineNum      = _lineNum;
        _tmpFrmVirtual.TransDate    = _transDate;
        _tmpFrmVirtual.Qty          = _qty;
        _tmpFrmVirtual.write();
    }
    ;

    // if we want to copy with header.
    //writeTmpFrmVirtual(tmpFrmVirtualHeader,
    //                   purchTableFrom.TableId,
    //                   purchTableFrom.RecId,
    //                   purchTableFrom.PurchId);
    //tmpFrmVirtualHeaderList.addEnd(tmpFrmVirtualHeader);
    while select purchLineFrom
        where purchLineFrom.PurchId == purchTableFrom.PurchId
    {
        tmpFrmVirtualLines.clear();
        tmpFrmVirtualLines.initValue();
        writeTmpFrmVirtual(tmpFrmVirtualLines,
                           purchLineFrom.TableId,
                           purchLineFrom.RecId,
                           purchLineFrom.PurchId,
                           purchLineFrom.LineNumber,
                           systemdateget(),
                           purchLineFrom.PurchQty);
        tmpFrmVirtualLinesList.addEnd(tmpFrmVirtualLines);
    }

    contract = PurchCopyingPurchTableContract::construct();   
    contract.parmCallingTable(purchTableTo);
    contract.parmPackedTmpFrmVirtualLines(tmpFrmVirtualLinesList.pack());
    contract.parmPackedTmpFrmVirtualHeader(tmpFrmVirtualHeaderList.pack());
    contract.parmQtyFactor(qtyFactor);
    contract.parmReverseSign(reverseSign);
    contract.parmRecalculateAmount(recalculateAmount);
    contract.parmCopyMarkup(copyMarkup);
    contract.parmCopyPrecisely(copyPrecisely);
    contract.parmDeleteLines(deleteLines);
    purchCopying.initParameters(contract);
    purchCopying.copy(); // transaction will created inside method
}