Добавление своих полей в форму заказа [обновлено]

При разработке нескольких проектов, возникала необходимость в получении дополнительных данных от покупателей, а полей в miniShop2 ограниченное количество. Поиск готового решения результата не дал, поэтому предлагаю свой вариант.

Решение обновлено, убраны правки исходного кода минишопа, теперь при обновлении ничего не затрется, изменены ключи у полей

Решалось это следующим образом:
1. Добавлялись необходимые поля, для примера взяты тип плательщика, название организации и инн.
2. Добавлялся плагин срабатывающий при сохранении заказа и при подключении js минишопа в админке.
3. Редактировались настройки и записи словарей.

Более подробно далее

1. В форму оформления заказа добавляем нужные поля:
<label><input type="radio" name="extfld_type" value="Юридическое лицо">Юридическое лицо</label>
<label><input type="radio" name="extfld_type" value="Физическое лицо">Физическое лицо</label>
<label><input type="text" name="extfld_org">Название организации</label>
<label><input type="text" name="extfld_inn">ИНН</label>
Все поля с префиксом extfld_ попадут в заказ.

2. Создаем плагин msExtraAddressFields и вешаем его на события msOnBeforeCreateOrder и msOnManagerCustomCssJs. Плагин сохраняет нужные поля при оформлении заказа в json в поле properties таблицы modx_ms2_order_addresses, а так же подключает нужный js в админке.
<?php
switch ($modx->event->name) {
    case 'msOnBeforeCreateOrder':
        $address = $msOrder->getOne('Address');
        $properties = array();
        foreach ($_POST as $key => $value){
            if (strpos($key,'extfld_') !== false){
                $properties[$key] = htmlentities($value,ENT_COMPAT | ENT_HTML401,'UTF-8');
            }
        }
        if (count($properties) > 0){
            $address->set('properties', json_encode($properties));    
        }
    break;
    
    case 'msOnManagerCustomCssJs':
        if ($page != 'orders') return;
	$modx->controller->addHtml("
            <script type='text/javascript'>
                Ext.ComponentMgr.onAvailable('minishop2-window-order-update', function(){
                	if (miniShop2.config['order_address_fields'].in_array('properties')){
                		if (this.record.addr_properties){
                		    var key;
                			for (key in this.record.addr_properties) {
                				this.fields.items[2].items.push(
                					{
                						xtype: 'displayfield',
                						name: 'addr_properties_'+key,
                						fieldLabel: _('ms2_properties_'+key),
                						anchor: '100%',
                						style: 'border:1px solid #efefef;width:95%;padding:5px;',
                						html: this.record.addr_properties[key]
                					}
                				);
                			}
                		}		
                	}
                });                
            </script>");
    break;
}

3. Добавляем вывод поля properties:
Заходим в системные настройки, там выбираем minishop2, блок Заказы.
Добавляем к значению «Поля адреса доставки» properties.


Добавляем записи словарей:
Заходим в Управление словарями, выбираем пространство имен minishop2, тема — manager, язык — ru.
Создаем новые записи с именами вида: ms2_properties_имя_вашего_поля (в данном случае ms2_properties_extfld_type, ms2_properties_extfld_org, ms2_properties_extfld_inn), значения — названия полей, которые будут отображаться в админке.

Поля не редактируемые, только выводят отправленные покупателем данные

Итоговый результат:

Дмитрий Меркурьев
11 октября 2016, 16:38
75
7 592
+11

Комментарии: 34

Павел Гвоздь
11 октября 2016, 20:42
+1
Сорцы править не хорошо.
    Дмитрий Меркурьев
    11 октября 2016, 21:30
    1
    0
    Согласен, и предупредил в посте о перезаписи при обновлении.
    Но все же лучше иметь хотя бы такое решение, чем не иметь его вообще. Каждый сам взвесит все плюсы и минусы и решит стоит ли использовать данный подход.
    А я в свою очередь постараюсь найти способ использования без правки исходного кода. Если есть мысли в какую сторону копать, то буду рад услышать.
Дмитрий Меркурьев
12 октября 2016, 14:52
+2
Решение обновлено, убраны правки исходного кода минишопа, теперь при обновлении ничего не затрется, изменены ключи у полей
    Марина Шипилова
    11 февраля 2017, 03:49
    0
    Здравствуйте. Все делаю, как вы написали. Но почему-то когда добавляю extfld_ в name — форма вообще не хочет отправлятся. Ругается, что ошибки в полях, хотя их нет. Подскажите, в чем может быть проблема?
Марат Марабар
12 октября 2016, 15:15
0
Чтобы js не подключался на каждой странице админки, я бы добавил ещё проверку
...
...
case 'msOnManagerCustomCssJs':
	if ($_GET['a'] != 'mgr/orders' && $_GET['namespace'] != 'minishop2') {
		return;
	}
...
...
Алексей
20 октября 2016, 12:22
0
Большое спасибо! все работает, только страничка заказов в админке не отображает заказы при первой загрузке (если нажать «обновить» в табличке -грузит ок.) и в консоль пишет:
ms2.utils.js?v=2.4.4-pl:13 Uncaught TypeError: date.strftime is not a function
вот этот файлик:
github.com/bezumkin/miniShop2/blob/master/assets/components/minishop2/js/mgr/misc/ms2.utils.js#L13
странно, но почемуто в последнем chrome видимо нету функции date.strftime?
    Дмитрий Меркурьев
    20 октября 2016, 12:35
    0
    В плагине не используются функции и методы из ms2.utils.js, так что скорее всего ошибка не из за него, при отключении плагина ошибка уходит и все работает нормально?
      Алексей
      20 октября 2016, 12:52
      0
      Нет, ошибка остается. При чем странно — при перезагрузке странички заказов в админке то все ок, то появляется эта ошибка, то нет. Никакой четкой зависимости не прослеживается. Вот скрин:

      и табличка заказов пуста…
        Дмитрий Меркурьев
        20 октября 2016, 13:09
        0
        Скорее всего какая то из дат приходит в неправильном формате и функции miniShop2.utils.formatDate не удается отформатировать ее в нужный вид. Я бы отправил в консоль string в этой функции и посмотреть что будет выдавать в случае появления ошибки.
Oracul22
21 октября 2016, 09:36
0
А как тогда значение этих доп полей вывести в чанке письма менеджерам?
Следующий способ не срабатывает:
[[+address.extfld_org]]
и этот тоже не срабатывает
[[+extfld_org]]
    Дмитрий Меркурьев
    21 октября 2016, 11:47
    +1
    Если используется Fenom то:
    {$address.properties.extfld_org}
    Если нет то:
    [[+address.properties.extfld_org]]
      Oracul22
      21 октября 2016, 12:43
      0
      Спасибо
      Андрей
      22 декабря 2016, 10:59
      0
      А скиньте пожалуйста свой чанк для отправки менеджеру с использованием Fenom, чтобы все поля приходили (тип оплаты, доставка и т.д.), а то не могу разобраться…
Максим
22 ноября 2016, 19:35
0
Странно, что вот это решение с сайта bezumkin.ru больше не доступно в интернете:
web.archive.org/web/20150419200400/https://bezumkin.ru/modx/minishop2/solutions/2052/

Или оно больше не работает?
    ck
    ck
    14 февраля 2017, 00:55
    0
    Вот здесь ещё предлагают вариант решения.
    Так всё-так как лучше добавить дополнительные поля к форме заказа, чтобы вывести их и на сайте и в админке MS2?
      Максим
      14 февраля 2017, 06:15
      0
      Не знаю, я делал как по ссылке, которую скидывал выше
Максим
24 ноября 2016, 21:00
0
А зачем вообще изначально в таблице modx_ms2_oerder_address поле properites? Что туда записывается?
Олег Щавелев
20 января 2017, 23:45
+1
Cпасибо, большое за плагин. Он классный.
Но у меня вопрос, а как можно добавить поле, в профайл приложения Office? Если есть решение, то вообще плагин супер классный)
uzunhair
09 января 2018, 19:20
0
А как сделать обязательными чекбокс? Установил в админке обязательные поля, обычные новые поля не позволяют оформить заказ пока пусты, а вот чекбокс проходит в любом случае.
<label class="form-check-label">
    <input class="form-check-input" type="checkbox" value="0" name="extfld_person" id="extfld_person">
    C условиями обработки персональных данных ознакомлен
</label>
    Рафис
    27 июня 2018, 15:59
    0
    Просто добавь required:
    <label class="form-check-label">
        <input class="form-check-input" type="checkbox" value="0" name="extfld_person" id="extfld_person" required>
        C условиями обработки персональных данных ознакомлен
    </label>
    Работает во всех современных браузерах. Но только для html5!
Рафис
27 июня 2018, 16:00
0
Спасибо большое за плагин)
Рафис
12 июля 2018, 15:20
0
1
Max Roganov
11 октября 2018, 16:52
0
Всем привет, а плагин еще актуален для самой свежей версии minishop и modx?
Все сделал по гайду правильно но поле в админке не выводится и ошибок в js консоли нет…
    Павел Романов
    11 октября 2018, 19:14
    +1
    Да, актуален, работает исправно.
    Проверьте имена полей и префикс.
    Илья
    09 февраля 2019, 21:21
    0
    разобрался? у меня тоже не работает :(
    Дмитрий Мансуров
    15 апреля 2019, 22:24
    0
    В системных настройках в поле
    ms2_order_address_fields
    проверь чтобы в списке значилось
    properties
Вася
24 декабря 2018, 16:04
0
А есть где то туториал как добавить редактируемые?
Игорь Терентьев
04 февраля 2019, 07:50
0
Не нужно ли проверить нет ли свойств в $properties, перед тем, как объявлять массив?
case 'msOnBeforeCreateOrder':
        $address = $msOrder->getOne('Address');
        $properties = $address->get('properties');
        if (!is_array($properties)) {
            $properties = array();
        }
        ...
Илья Александрович
14 февраля 2019, 11:22
0
Скажите, а как вытащить свои поля? Мне нужно их передать в плагин.
$this->Order->Address->get('extfld_ * ')
— как то так пробовал, не получилось
Авторизуйтесь или зарегистрируйтесь, чтобы оставлять комментарии.