Практические примеры#

В этом разделе рассмотрены два практических сценария: автоматическое создание пунктов маршрута для складов и обработка перехода состояний документов с валидацией бизнес-правил.

Создание пункта маршрута для каждого склада#

1. SQL-запрос для получения списка складов. Получим ID и коды складов, у которых тип объекта «Склад» (stock) или «Склад в пути» (StockInWay). Полученные записи преобразуем в список с помощью метода asList().

var sqlStock = sql(`
    SELECT stk.id as id, stk.sCode as sCode FROM stk_stock stk
    JOIN btk_objecttype ot ON ot.id = stk.idObjectType
    where ot.sCode = 'stock' OR ot.sCode = 'StockInWay'
`).asList()

2. Поиск пункта маршрута по коду склада. С помощью цикла for сделаем преобразование над полученными записями. По коду нашего склада получим id пункта маршрута.

var sqlStock = sql(`
    SELECT stk.id as id, stk.sCode as sCode FROM stk_stock stk
    JOIN btk_objecttype ot ON ot.id = stk.idObjectType
    where ot.sCode = 'stock' OR ot.sCode = 'StockInWay'
`).asList()

for(r: sqlStock) {
    var idvPoint = Tms_PointApi.findByMnemoCode(r.scode);
}

3. Создание нового пункта маршрута (если не найден). Если вернувшееся значение пункта маршрута равно null, то нужно создать его, заполнив код, название и привязку текущего склада.

var sqlStock = sql(`
    SELECT stk.id as id, stk.sCode as sCode FROM stk_stock stk
    JOIN btk_objecttype ot ON ot.id = stk.idObjectType
    where ot.sCode = 'stock' OR ot.sCode = 'StockInWay'
`).asList()

for(r: sqlStock) {
    var idvPoint = Tms_PointApi.findByMnemoCode(r.scode);
    if (isNull(idvPoint)) {
        var ropPoint = Tms_PointApi.insert();
        Tms_PointApi.setsCode(ropPoint, r.scode);
        Tms_PointApi.setsCaption(ropPoint, r.scode);
        Tms_PointApi.setidStock(ropPoint, r.id);
    }
}

4. Обновление существующего пункта маршрута (если не привязан к складу). Если полученный пункт маршрута существует, то загружаем его по полученному ранее id. Далее проверяем у полученного пункта маршрута привязку к складу, если значение равно null, то устанавливаем значение текущего склада.

var sqlStock = sql(`
    SELECT stk.id as id, stk.sCode as sCode FROM stk_stock stk
    JOIN btk_objecttype ot ON ot.id = stk.idObjectType
    where ot.sCode = 'stock' OR ot.sCode = 'StockInWay'
`).asList()

for(r: sqlStock) {
    var idvPoint = Tms_PointApi.findByMnemoCode(r.scode);
    if (isNull(idvPoint)) {
        var ropPoint = Tms_PointApi.insert();
        Tms_PointApi.setsCode(ropPoint, r.scode);
        Tms_PointApi.setsCaption(ropPoint, r.scode);
        Tms_PointApi.setidStock(ropPoint, r.id);
    } else {
        var ropPointW = Tms_PointApi.load(idvPoint);
        if (isNull(ropPointW.idStock)) {
            Tms_PointApi.setidStock(ropPointW, r.id);
        }
    }
}

Пример расширения обработки состояния документа#

1. Проверка на пустое старое значение:

if(isNull(oldValue)) return 0;

Если текущее состояние документа (oldValue) не задано, обработка прекращается (возвращается 0).

2. Получение порядковых номеров состояний:

var idStateFromOrder = Btk_ClassStateApi.getOrder(oldValue);
var idStateToOrder = Btk_ClassStateApi.getOrder(value);
  • idStateFromOrder - порядковый номер старого состояния документа;

  • idStateToOrder - порядковый номер нового состояния.

Получим текущее состояние документа, воспользовавшись API-методом класса состояния.

3. Проверка условий для обработки:

if(idStateFromOrder <= 100 && idStateToOrder > 100 && rop.idObjectType == Btk_ObjectTypeApi.findByMnemoCode("Tms_DeliveryAct"))
  • idStateFromOrder <= 100 - проверяет, что текущее состояние документа имеет порядковый номер ≤ 100. Обычно это соответствует статусам «Аннулировано» или «Формируется».

  • idStateToOrder > idStateFromOrder - гарантирует, что новое состояние действительно является переходом ВВЕРХ по workflow. Исключает случаи, когда статус меняется на равный или более ранний.

  • rop.idObjectType == Btk_ObjectTypeApi.findByMnemoCode(«Tms_DeliveryAct») - убеждается, что работаем именно с документом типа «Рейс» (Tms_DeliveryAct).

4. Получение заказа (Stm_OrderOut):

var ropOO = Stm_OrderOutApi.loadByGid(rop.gidSrc);
if(ropOO.idObjectType != Btk_ObjectTypeApi.findByMnemoCode("OrderOut_Service"))

Получаем ропу заказа с помощью Api-метода loadByGid по gidSrc(родитель) текущего документа. Если тип заказа НЕ «OrderOut_Service» (то есть это не сервисный заказ), выполняется дальнейшая проверка.

5. Определение договора:

var ropContract = null;
if(isNotNull(rop.idAddContract))
    ropContract = Cnt_ContractApi.load(rop.idAddContract);
else
    ropContract = Cnt_ContractApi.load(rop.idContract);

Если у заказа (rop) есть доп. договор (idAddContract), он загружается. Иначе загружается основной договор (idContract).

6. Проверка отложенного ценообразования:

var bDeferredPricing = Cnt_ContractApi.getObjAttrValue(ropContract, "bDeferredPricing");
if(bDeferredPricing == true)

Получаем значение атрибута bDeferredPricing (флаг «Отложенное ценообразование»). Если флаг true, выполняется проверка даты.

7. Проверка даты отложенного ценообразования:

var dDeferredPricing = Stm_ActOutApi.getObjAttrValue(rop, "dDeferredPricing");
if(isNull(dDeferredPricing) || dDeferredPricing < sysDate())
    raise("Требуется выполнить корректировку цены");

Получаем дату dDeferredPricing (когда должна быть выполнена корректировка цены). Если дата не задана или уже прошла, выводится ошибка.

Общий код:

if(isNull(oldValue))return 0;
var idStateFromOrder = Btk_ClassStateApi.getOrder(oldValue)
var idStateToOrder = Btk_ClassStateApi.getOrder(value);

if(idStateFromOrder <= 100 && idStateToOrder > 100 && rop.idObjectType == Btk_ObjectTypeApi.findByMnemoCode("Tms_DeliveryAct")){
    var ropOO = Stm_OrderOutApi.loadByGid(rop.gidSrc)
    if(ropOO.idObjectType != Btk_ObjectTypeApi.findByMnemoCode("OrderOut_Service")){
        var ropContract = null
        if(isNotNull(rop.idAddContract))
            ropContract = Cnt_ContractApi.load(rop.idAddContract)
        else
            ropContract = Cnt_ContractApi.load(rop.idContract)
        var bDeferredPricing = Cnt_ContractApi.getObjAttrValue(ropContract, "bDeferredPricing")
        if(bDeferredPricing == true){
            var dDeferredPricing = Stm_ActOutApi.getObjAttrValue(rop, "dDeferredPricing")
            if(isNull(dDeferredPricing) || dDeferredPricing < sysDate()){
                raise("Требуется выполнить корректировку цены")
            }
        }
    }
}