AXForum  
Вернуться   AXForum > Microsoft Dynamics AX > DAX: Программирование
All
Забыли пароль?
Зарегистрироваться Правила Справка Пользователи Сообщения за день Поиск Все разделы прочитаны

 
 
Опции темы Поиск в этой теме Опции просмотра
Старый 30.07.2009, 16:52   #1  
Lanai is offline
Lanai
Участник
 
35 / 29 (1) +++
Регистрация: 09.08.2005
validateField(),leave(),modified()
Задача простая с первого взгляда (или на самом деле простая):

Необходимо чтобы при изменении некоторых полей в форме запрашивался комментарий (причина изменения) и сохранялась история.

Знаю, что вариантов много, но все они не подходят по разным причинам. Хочется оптимально правильный вариант найти. Спасибо!
Старый 30.07.2009, 17:12   #2  
lev is offline
lev
Ищущий знания...
Аватар для lev
Oracle
MCBMSS
Axapta Retail User
 
1,723 / 491 (20) +++++++
Регистрация: 18.01.2005
Адрес: Москва
Как вариант:
На таблице (не на форме, а именно на самой таблице) есть метод ModifiedField(), в котором можно перехватывать изменение полей, и выполнять определенные действия.
__________________
"Страх перед возможностью ошибки не должен отвращать нас от поисков истины." (с)
С Уважением,
Елизаров Артем
Старый 30.07.2009, 17:13   #3  
ZVV is offline
ZVV
MCITP
MCP
Oracle
MCBMSS
 
1,006 / 246 (11) ++++++
Регистрация: 13.02.2004
Адрес: Минск
->
Цитата:
Сообщение от Lanai Посмотреть сообщение
Задача простая с первого взгляда (или на самом деле простая):

Необходимо чтобы при изменении некоторых полей в форме запрашивался комментарий (причина изменения) и сохранялась история.

Знаю, что вариантов много, но все они не подходят по разным причинам. Хочется оптимально правильный вариант найти. Спасибо!
Как вариант:
- Добавляете в таблицу поле "коментарий"...
- "при изменении некоторых полей в форме" требуете чтоб этот комментарий был заполнен (обязательный)
- при сохранении строки (на update(), например) сохраняете историю изменения с комментарием и чистите поле "комментарий" в целевой таблице.

PS Для реализации второго пункта можно инспользовать .orig() и validateWrite()...
__________________
Zhirenkov Vitaly

Последний раз редактировалось ZVV; 30.07.2009 в 17:16.
Старый 30.07.2009, 17:16   #4  
belugin is offline
belugin
Участник
Аватар для belugin
Сотрудники Microsoft Dynamics
Лучший по профессии 2017
Лучший по профессии 2015
Лучший по профессии 2014
Лучший по профессии 2011
Лучший по профессии 2009
 
4,622 / 2925 (107) +++++++++
Регистрация: 16.01.2004
Записей в блоге: 5
еще вариант - в update датасурса перед супером сравнивать этих значения полей с предыдущими (orig) и запрашивать коммент на все сразу
Старый 30.07.2009, 17:18   #5  
DSPIC is offline
DSPIC
Боец
 
1,077 / 1243 (44) ++++++++
Регистрация: 11.04.2008
1. Если это нужно сделать для конкретной формы, то лучше этот функционал написать на форме (а ещё лучше в классе и вызывать из формы). Если это нужно для таблицы, которая светится на нескольких формах, то лучше этот функционал написать на таблице, но при этом проверять, что вызов произошел из нужной формы, т.к. изменение полей таблицы может происходить не только по явной инициативе пользователя.
2. В примере данном примере вызов происходит на датасорсе формы Address, метод validateWrite():

X++:
public boolean validateWrite()
{
    boolean ret;
    Tutorial_RunbaseForm    tutorial_RunbaseForm;
    ;

    ret = super();

    if (ret && Address.RecId)
    {
        if (Address.Name != Address.orig().Name)
        {
            tutorial_RunbaseForm = Tutorial_RunbaseForm::construct();

            if (tutorial_RunbaseForm.prompt())
            {
                tutorial_RunbaseForm.run();
            }
            else
            {
                Address.reread();
                ret = checkfailed("ПОля не сохранены");
            }
        }
    }

    return ret;
}
Здесь проверяется на изменение поле "Name".
В качестве диалога\формы для указания причины изменения подставил диалог tutorial_RunbaseForm. У вас должен быть свой.

3. Это первый пример, который пришел в голову, соотв-нно на красивость и полную корректность не претендует.
Старый 30.07.2009, 17:20   #6  
DSPIC is offline
DSPIC
Боец
 
1,077 / 1243 (44) ++++++++
Регистрация: 11.04.2008
Цитата:
Сообщение от belugin Посмотреть сообщение
еще вариант - в update датасурса перед супером сравнивать этих значения полей с предыдущими (orig) и запрашивать коммент на все сразу
Нет таких методов на датасорсе Есть write(), который по RecId условно расчепляется на update и insert
Старый 30.07.2009, 17:25   #7  
ZVV is offline
ZVV
MCITP
MCP
Oracle
MCBMSS
 
1,006 / 246 (11) ++++++
Регистрация: 13.02.2004
Адрес: Минск
->
Цитата:
Сообщение от DSPIC Посмотреть сообщение
1. Если это нужно сделать для конкретной формы, то лучше этот функционал написать на форме (а ещё лучше в классе и вызывать из формы). Если это нужно для таблицы, которая светится на нескольких формах, то лучше этот функционал написать на таблице, но при этом проверять, что вызов произошел из нужной формы, т.к. изменение полей таблицы может происходить не только по явной инициативе пользователя.
2. В примере данном примере вызов происходит на датасорсе формы Address, метод validateWrite():

X++:
public boolean validateWrite()
{
    boolean ret;
    Tutorial_RunbaseForm    tutorial_RunbaseForm;
    ;

    ret = super();

    if (ret && Address.RecId)
    {
        if (Address.Name != Address.orig().Name)
        {
            tutorial_RunbaseForm = Tutorial_RunbaseForm::construct();

            if (tutorial_RunbaseForm.prompt())
            {
                tutorial_RunbaseForm.run();
            }
            else
            {
                Address.reread();
                ret = checkfailed("ПОля не сохранены");
            }
        }
    }

    return ret;
}
Здесь проверяется на изменение поле "Name".
В качестве диалога\формы для указания причины изменения подставил диалог tutorial_RunbaseForm. У вас должен быть свой.

3. Это первый пример, который пришел в голову, соотв-нно на красивость и полную корректность не претендует.
Всё бы хорошо, если б не одно "НО".
validateWrite() - идёт без транзакции, а сам update() потом пойдёт уже отдельной транзакцией
Соответсвенно потенциально имете проблему, когда лог вставился, а изменения не прошли.

Вариант belugin-а может и лишён конкретно этого недостатка (чего уже придираться то, кстати, - понятно же что он имел ввиду ), но в этом случае будет выскакивать запрос пользователю во время выполнения транзакции, что, я думаю все понимают, не есть гут..
__________________
Zhirenkov Vitaly
Старый 30.07.2009, 17:27   #8  
petr is offline
petr
Участник
Соотечественники
 
561 / 201 (8) ++++++
Регистрация: 30.05.2005
Адрес: Швейцария
Цитата:
Сообщение от belugin Посмотреть сообщение
еще вариант - в update датасурса перед супером сравнивать этих значения полей с предыдущими (orig) и запрашивать коммент на все сразу
Тут только надо быть осторожным. Если это write на датасорсе формы то все ок, если же метод update на таблице, то надо не забыть проверить текущий уровень транзакции, иначе при обновлении таблицы из кода внутри транзакции будет запущена форма для взаимодейсвия с пользователем, что категорически нельзя допускать.

Я однажды такую ошибку допускил и меня EVGL за это страшно отругал

Не повторяйте чужих ошибок...
Старый 30.07.2009, 17:27   #9  
belugin is offline
belugin
Участник
Аватар для belugin
Сотрудники Microsoft Dynamics
Лучший по профессии 2017
Лучший по профессии 2015
Лучший по профессии 2014
Лучший по профессии 2011
Лучший по профессии 2009
 
4,622 / 2925 (107) +++++++++
Регистрация: 16.01.2004
Записей в блоге: 5
Надо делать двойную проверку: сначала что изменения есть (и выдать сообщение). Потом в транзакции что нет некомменированных изменений
Старый 30.07.2009, 17:36   #10  
belugin is offline
belugin
Участник
Аватар для belugin
Сотрудники Microsoft Dynamics
Лучший по профессии 2017
Лучший по профессии 2015
Лучший по профессии 2014
Лучший по профессии 2011
Лучший по профессии 2009
 
4,622 / 2925 (107) +++++++++
Регистрация: 16.01.2004
Записей в блоге: 5
Цитата:
Сообщение от petr Посмотреть сообщение
Тут только надо быть осторожным. Если это write на датасорсе формы
я имелл виду write на датасурсе формы.
Старый 30.07.2009, 17:41   #11  
Lanai is offline
Lanai
Участник
 
35 / 29 (1) +++
Регистрация: 09.08.2005
2 belugin: Коммент должен быть к каждому полю свой.

2 lev: Забыл сказать версия 2.5. В 2.5 нет метода modifiedField() на таблице! Даже если бы и был, то не ясно какие именно действия нужно выполнить.

2 ZVV: Такой вариант допустим, но не хотелось бы например в справочнике клиентов создавать какое-то поле для временного хранения комментов - как-то не очень красиво. Но вариант!

2 DSPIC: Ща попробую - отпишу.

Всем спасибо за ответы!
Старый 30.07.2009, 17:55   #12  
Lanai is offline
Lanai
Участник
 
35 / 29 (1) +++
Регистрация: 09.08.2005
2 DSPIC: Попробовал! Во-первых код какой-то слишком большой получается (т.к. полей, которые нужно проверять около 15) - да и как-то не оптимально каждый раз проверять изменилось ли что-либо из них, но самое главное что validateWrite() вызывается один раз для всех полей - если например использовать для ввода комментария диалог, то после сохранения изменений в строке выскакивает сразу друг за другом несколько (столько, сколько полей мы изменили) диалоговых окон и не ясно куда какие комментарии вводить. Хотелось бы, чтобы запрос комментария выскакивал сразу после изменения поля (каждому полю свой).
Старый 30.07.2009, 18:08   #13  
DSPIC is offline
DSPIC
Боец
 
1,077 / 1243 (44) ++++++++
Регистрация: 11.04.2008
Цитата:
Хотелось бы, чтобы запрос комментария выскакивал сразу после изменения поля (каждому полю свой).
А что, если юзер поменял, скажем, 5 полей, вбил причины и потом, не сохранив запись, передумал и нажал "Esc"? Думаю, так некорректно.
Тогда как предложение с т.з. пользователя:
При сохранении записи вываливается список модифицированных полей: форма(диалог) с гридом и полями в нём:
Имя поля Причина модификации

+ Кнопки OK\Cancel
Примерно так. Реализация на порядок сложнее, но решение можно сделать универсальным для любой таблицы...
Старый 30.07.2009, 18:10   #14  
belugin is offline
belugin
Участник
Аватар для belugin
Сотрудники Microsoft Dynamics
Лучший по профессии 2017
Лучший по профессии 2015
Лучший по профессии 2014
Лучший по профессии 2011
Лучший по профессии 2009
 
4,622 / 2925 (107) +++++++++
Регистрация: 16.01.2004
Записей в блоге: 5
Цитата:
Сообщение от Lanai Посмотреть сообщение
2 DSPIC: Попробовал! Во-первых код какой-то слишком большой получается (т.к. полей, которые нужно проверять около 15)
X++:
container loggedFields = [
      fieldNum(T1, F1), fieldNum(T1, F2) ...
];

for(i=1; i<=conLen(loggedFields); i++)
{
    fieldId = conPeek(loggedFields, i)
    if (record.(fieldId)!=record.orig().(fieldId))
        changedFields +=  fieldId;
}
Цитата:
- да и как-то не оптимально каждый раз проверять изменилось ли что-либо из них, но самое главное что validateWrite() вызывается один раз для всех полей - если например использовать для ввода комментария диалог, то после сохранения изменений в строке выскакивает сразу друг за другом несколько (столько, сколько полей мы изменили) диалоговых окон
X++:
Dialog d = new Dialog('comnnmants');
for(i=1;i<=conLen(changedFields);i++)
{
    d.addField(typeId(Commant), fieldId2PName(record.table, conPeek(fields, i));
}
А можно спрашивать после каждого изменения и хранить map<recid, map<filedID, comment>> и сбрасывать его после сохранения.

Можно добавить новый таб с гридом с комментами по изменениями (хранить во временной таблице)
Старый 30.07.2009, 18:11   #15  
belugin is offline
belugin
Участник
Аватар для belugin
Сотрудники Microsoft Dynamics
Лучший по профессии 2017
Лучший по профессии 2015
Лучший по профессии 2014
Лучший по профессии 2011
Лучший по профессии 2009
 
4,622 / 2925 (107) +++++++++
Регистрация: 16.01.2004
Записей в блоге: 5
Цитата:
Сообщение от DSPIC Посмотреть сообщение
А что, если юзер поменял, скажем, 5 полей, вбил причины и потом, не сохранив запись, передумал и нажал "Esc"?
Разумеется созхраняться комменты должны при сохранении записи а до этого где-то временно храниться

Правда тут надо озаботиться чтобы переход по записям не приводил к сохранению комментов на неправильной записи и т,д, _ вобщем тут я бы обязательно написал чеклист того, как это тестировать
Старый 30.07.2009, 18:19   #16  
DSPIC is offline
DSPIC
Боец
 
1,077 / 1243 (44) ++++++++
Регистрация: 11.04.2008
Цитата:
Сообщение от belugin Посмотреть сообщение
[XPP]container loggedFields = [
fieldNum(T1, F1), fieldNum(T1, F2) ...
];

for(i=1; i<=conLen(loggedFields); i++)
{
fieldId = conPeek(loggedFields, i)
if (record.(fieldId)!=record.orig().(fieldId))
changedFields += fieldId;
}
Да, в этом русле, только для универсальности, loggedFields сделать не хардкодом в контэйнере, а создать на таблице группу и добавлять туда поля для логирования. Тогда решение будет легко мигрироваться а список полей для логирования легко кастомизироваться.
Старый 30.07.2009, 20:48   #17  
Андре is offline
Андре
Moderator
Сотрудники компании GMCS
 
2,375 / 464 (20) +++++++
Регистрация: 03.12.2001
Как вариант - доработать стандартное журналирование данных системы, добавив в журнал одно поле - 'примечание'. Не уверен, что оттуда получится выводить диалоговое окно для его ввода, но посмотреть в этом направлении можно.

Мы таким образом в журнал БД добавляли стек вызовов, изменивший данную запись в таблице.
Теги
запись, как правильно, комментарий

 

Похожие темы
Тема Автор Раздел Ответов Посл. сообщение
axperf: Create RecID index on tables with Created/Modified DateTime fields Blog bot DAX Blogs 0 20.06.2009 10:05
modified, leave Varmen DAX: Программирование 7 19.03.2009 13:51
Вызов метода modified DS Mike2005 DAX: Программирование 1 17.10.2006 13:45
modified два раза два раза MironovI DAX: Программирование 5 14.07.2006 16:29
Транзакция в modified Gad DAX: Программирование 1 29.11.2005 19:54
Опции темы Поиск в этой теме
Поиск в этой теме:

Расширенный поиск
Опции просмотра

Ваши права в разделе
Вы не можете создавать новые темы
Вы не можете отвечать в темах
Вы не можете прикреплять вложения
Вы не можете редактировать свои сообщения

BB коды Вкл.
Смайлы Вкл.
[IMG] код Вкл.
HTML код Выкл.
Быстрый переход

Рейтинг@Mail.ru
Часовой пояс GMT +3, время: 14:33.