Compare commits
20 commits
Author | SHA1 | Date | |
---|---|---|---|
|
e61259ded9 | ||
|
0359315c79 | ||
|
08ff21ce04 | ||
|
de7ecd4100 | ||
|
54692c50d4 | ||
|
61044988d5 | ||
|
7a36fa5de7 | ||
|
8cf4420592 | ||
|
84c7d10146 | ||
|
8c38d21477 | ||
|
115cf33423 | ||
|
3162031b65 | ||
|
1ec3bccd0c | ||
|
41257b9d7c | ||
|
b68779a054 | ||
|
d25c7fc484 | ||
|
317a30b682 | ||
|
534757e6c6 | ||
|
7e260a2d22 | ||
|
6fb061e6cc |
32 changed files with 1234 additions and 1379 deletions
2
.github/workflows/ci.yml
vendored
2
.github/workflows/ci.yml
vendored
|
@ -49,7 +49,7 @@ jobs:
|
|||
run: composer validate
|
||||
- name: Cache Composer packages
|
||||
id: composer-cache
|
||||
uses: actions/cache@v2
|
||||
uses: actions/cache@v3
|
||||
with:
|
||||
path: vendor
|
||||
key: ${{ runner.os }}-php-${{ hashFiles('**/composer.lock') }}
|
||||
|
|
58
CHANGELOG.md
58
CHANGELOG.md
|
@ -1,3 +1,61 @@
|
|||
|
||||
## 2025-03-26 v6.6.11
|
||||
- Исправлена передача габаритов при выгрузке заказов по агенту
|
||||
|
||||
## 2025-03-25 v6.6.10
|
||||
- Исправлено некорректное изменение статуса оплаты отмененных заказов
|
||||
|
||||
## 2025-03-04 v6.6.9
|
||||
- Исправлено обновление модуля
|
||||
|
||||
## 2025-03-03 v6.6.8
|
||||
- Исправлена ошибка экспорта дополнительных свойств товаров
|
||||
|
||||
## 2025-02-04 v6.6.7
|
||||
- Исправлена ошибка установки модуля на PHP 8.0
|
||||
|
||||
## 2025-01-29 v6.6.6
|
||||
- Поддержка нулевой закупочной стоимости при генерации каталога
|
||||
|
||||
## 2025-01-28 v6.6.5
|
||||
- Исправлена ошибка редактирования интеграционных доставок при активации опции передачи статусов интеграционных оплат
|
||||
|
||||
## 2025-01-27 v6.6.4
|
||||
- Исправлено некорректное удаление признака применения промокода при изменении состава заказа в CRM
|
||||
|
||||
## 2025-01-14 v6.6.3
|
||||
- Исправлены ошибки при обновлении модуля
|
||||
|
||||
## 2025-01-14 v6.6.2
|
||||
- Исправлена выгрузка архива заказов при установке модуля
|
||||
|
||||
## 2024-12-17 v6.6.1
|
||||
- Исправлены API методы по взаимодействию с пользовательскими полями и справочниками
|
||||
|
||||
## 2024-12-09 v6.6.0
|
||||
- Добавлено динамическое изменение свойств товаров при настройке экспорта
|
||||
|
||||
## 2024-12-08 v6.5.39
|
||||
- Исправлена поломка заказов с промокодом Maxma при включенной передаче корзины в CRM
|
||||
|
||||
## 2024-10-31 v6.5.38
|
||||
- Исправлена выгрузка заказов через агент
|
||||
|
||||
## 2024-10-30 v6.5.37
|
||||
- Исправлена подписка модуля на событие сохранения заказа
|
||||
|
||||
## 2024-10-24 v6.5.36
|
||||
- Добавлена передача профилей доставки Официального модуля Почты России
|
||||
|
||||
## 2024-10-22 v6.5.35
|
||||
- Исправлена подписка модуля на событие сохранения заказа
|
||||
|
||||
## 2024-10-21 v6.5.34
|
||||
- Исправлена ошибка агента выгрузки изменений при наличии у заказа клиента без магазина
|
||||
|
||||
## 2024-10-14 v6.5.33
|
||||
- Добавлена передача дополнительных параметров в GET запросах
|
||||
|
||||
## 2024-10-03 v6.5.32
|
||||
- Исправлена подписка на событие создания заказа при обновлении
|
||||
|
||||
|
|
|
@ -1508,7 +1508,7 @@ class ApiClient
|
|||
}
|
||||
|
||||
return $this->client->makeRequest(
|
||||
"/custom-fields/$entity/edit/{$customField['code']}",
|
||||
"/custom-fields/$entity/{$customField['code']}/edit",
|
||||
Client::METHOD_POST,
|
||||
array('customField' => json_encode($customField))
|
||||
);
|
||||
|
@ -1591,7 +1591,7 @@ class ApiClient
|
|||
}
|
||||
|
||||
return $this->client->makeRequest(
|
||||
"/custom-fields/dictionaries/{$customDictionary['code']}/create",
|
||||
"/custom-fields/dictionaries/create",
|
||||
Client::METHOD_POST,
|
||||
array('customDictionary' => json_encode($customDictionary))
|
||||
);
|
||||
|
|
|
@ -11,6 +11,7 @@
|
|||
|
||||
namespace RetailCrm\Http;
|
||||
|
||||
use Intaro\RetailCrm\Component\Constants;
|
||||
use RetailCrm\Exception\CurlException;
|
||||
use RetailCrm\Exception\InvalidJsonException;
|
||||
use RetailCrm\Response\ApiResponse;
|
||||
|
@ -38,7 +39,7 @@ class Client
|
|||
*
|
||||
* @throws \InvalidArgumentException
|
||||
*/
|
||||
public function __construct($url, array $defaultParameters = array())
|
||||
public function __construct($url, array $defaultParameters = [])
|
||||
{
|
||||
if (false === stripos($url, 'https://')) {
|
||||
throw new \InvalidArgumentException(
|
||||
|
@ -49,7 +50,7 @@ class Client
|
|||
$this->url = $url;
|
||||
$this->defaultParameters = $defaultParameters;
|
||||
$this->retry = 0;
|
||||
$this->curlErrors = array(
|
||||
$this->curlErrors = [
|
||||
CURLE_COULDNT_RESOLVE_PROXY,
|
||||
CURLE_COULDNT_RESOLVE_HOST,
|
||||
CURLE_COULDNT_CONNECT,
|
||||
|
@ -57,8 +58,8 @@ class Client
|
|||
CURLE_HTTP_POST_ERROR,
|
||||
CURLE_SSL_CONNECT_ERROR,
|
||||
CURLE_SEND_ERROR,
|
||||
CURLE_RECV_ERROR
|
||||
);
|
||||
CURLE_RECV_ERROR,
|
||||
];
|
||||
}
|
||||
|
||||
/**
|
||||
|
@ -79,9 +80,9 @@ class Client
|
|||
public function makeRequest(
|
||||
$path,
|
||||
$method,
|
||||
array $parameters = array()
|
||||
array $parameters = []
|
||||
) {
|
||||
$allowedMethods = array(self::METHOD_GET, self::METHOD_POST);
|
||||
$allowedMethods = [self::METHOD_GET, self::METHOD_POST];
|
||||
|
||||
if (!in_array($method, $allowedMethods, false)) {
|
||||
throw new \InvalidArgumentException(
|
||||
|
@ -93,7 +94,14 @@ class Client
|
|||
);
|
||||
}
|
||||
|
||||
$parameters = array_merge($this->defaultParameters, $parameters);
|
||||
$parameters = self::METHOD_GET === $method
|
||||
? array_merge($this->defaultParameters, $parameters, [
|
||||
'cms_source' => 'Bitrix',
|
||||
'cms_version' => SM_VERSION,
|
||||
'php_version' => function_exists('phpversion') ? phpversion() : '',
|
||||
'module_version' => Constants::MODULE_VERSION,
|
||||
])
|
||||
: $parameters = array_merge($this->defaultParameters, $parameters);
|
||||
|
||||
$url = $this->url . $path;
|
||||
|
||||
|
@ -122,7 +130,8 @@ class Client
|
|||
|
||||
curl_close($curlHandler);
|
||||
|
||||
if ($errno
|
||||
if (
|
||||
$errno
|
||||
&& in_array($errno, $this->curlErrors, false)
|
||||
&& $this->retry < 3
|
||||
) {
|
||||
|
|
|
@ -266,8 +266,8 @@ class RCrmActions
|
|||
if (is_array($failedIds) && !empty($failedIds)) {
|
||||
RetailCrmOrder::uploadOrders(50, true);
|
||||
}
|
||||
|
||||
return;
|
||||
|
||||
return 'RCrmActions::uploadOrdersAgent();';
|
||||
}
|
||||
|
||||
/**
|
||||
|
@ -290,7 +290,6 @@ class RCrmActions
|
|||
|
||||
RetailCrmHistory::customerHistory();
|
||||
RetailCrmHistory::orderHistory();
|
||||
self::uploadOrdersAgent();
|
||||
} catch (\Throwable $exception) {
|
||||
Logger::getInstance()->write(
|
||||
'Fail orderAgent:' . PHP_EOL .
|
||||
|
|
|
@ -144,7 +144,7 @@ class RetailCrmCart
|
|||
|
||||
|
||||
foreach ($items as $item) {
|
||||
$itemFields = $item->getFields();
|
||||
$itemFields = $item->getFields()->getValues();
|
||||
|
||||
if (isset($basketItems[(int) $itemFields['ID']])) {
|
||||
$itemFields['PRICE'] = $basketItems[(int) $itemFields['ID']]['PRICE'];
|
||||
|
|
|
@ -1015,6 +1015,14 @@ class RetailCrmHistory
|
|||
$order['site']
|
||||
);
|
||||
|
||||
if (!$response) {
|
||||
Logger::getInstance()->write(
|
||||
sprintf('Заказ %s не выгружен. Не удалось получить клиента', $order['id']),
|
||||
'orderAgent'
|
||||
);
|
||||
continue;
|
||||
}
|
||||
|
||||
$newUser = new CUser();
|
||||
$customerBuilder = new CustomerBuilder();
|
||||
$customerBuilder->setDataCrm($response['customer'])->build();
|
||||
|
@ -1218,7 +1226,6 @@ class RetailCrmHistory
|
|||
}
|
||||
|
||||
if (array_key_exists('discountTotal_sum', $collectItems[$product['offer']['externalId']])) {
|
||||
$item->setField('CUSTOM_PRICE', 'Y');
|
||||
$item->setField('DISCOUNT_NAME', '');
|
||||
$item->setField('DISCOUNT_VALUE', '');
|
||||
|
||||
|
@ -1232,6 +1239,7 @@ class RetailCrmHistory
|
|||
$price = self::truncate($price, 2);
|
||||
}
|
||||
|
||||
$item->markFieldCustom('PRICE');
|
||||
$item->setField('PRICE', $price);
|
||||
}
|
||||
|
||||
|
@ -2054,7 +2062,7 @@ class RetailCrmHistory
|
|||
$payment->delete();
|
||||
}
|
||||
|
||||
if ($paymentsCrm['totalSumm'] == $paySumm) {
|
||||
if ($paymentsCrm['totalSumm'] == $paySumm && $order->getField('CANCELED') !== 'Y') {
|
||||
$order->setFieldNoDemand('PAYED', 'Y');
|
||||
} else {
|
||||
$order->setFieldNoDemand('PAYED', 'N');
|
||||
|
|
|
@ -10,6 +10,7 @@
|
|||
*/
|
||||
|
||||
use Bitrix\Main\Context;
|
||||
use Bitrix\Main\Type\DateTime;
|
||||
use Bitrix\Main\Context\Culture;
|
||||
use Bitrix\Main\UserTable;
|
||||
use Bitrix\Sale\Delivery\Services\Manager;
|
||||
|
@ -357,7 +358,7 @@ class RetailCrmOrder
|
|||
|
||||
$order['items'][] = $item;
|
||||
|
||||
if ($send && $dimensionsSetting === 'Y') {
|
||||
if ($dimensionsSetting === 'Y') {
|
||||
$dimensions = RCrmActions::unserializeArrayRecursive($product['DIMENSIONS']);
|
||||
|
||||
if ($dimensions !== false) {
|
||||
|
@ -369,7 +370,7 @@ class RetailCrmOrder
|
|||
}
|
||||
}
|
||||
|
||||
if ($send && $dimensionsSetting === 'Y') {
|
||||
if ($dimensionsSetting === 'Y') {
|
||||
$order['width'] = $width;
|
||||
$order['height'] = $height;
|
||||
$order['length'] = $length;
|
||||
|
@ -429,7 +430,7 @@ class RetailCrmOrder
|
|||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
if ('ordersEdit' === $methodApi) {
|
||||
$order = RetailCrmService::unsetIntegrationDeliveryFields($order);
|
||||
}
|
||||
|
@ -536,9 +537,11 @@ class RetailCrmOrder
|
|||
$resCustomersAdded = [];
|
||||
$resCustomersCorporate = [];
|
||||
$orderIds = [];
|
||||
$orderUpdateIds = [];
|
||||
|
||||
$lastUpOrderId = RetailcrmConfigProvider::getLastOrderId();
|
||||
$failedIds = RetailcrmConfigProvider::getFailedOrdersIds();
|
||||
$uploadMethod = RetailcrmConfigProvider::getOrderDischarge();
|
||||
|
||||
if ($failed == true && $failedIds !== false && count($failedIds) > 0) {
|
||||
$orderIds = $failedIds;
|
||||
|
@ -557,6 +560,22 @@ class RetailCrmOrder
|
|||
}
|
||||
}
|
||||
|
||||
if((int)$uploadMethod === 0) {
|
||||
$dateOfLastUpdate = RetailcrmConfigProvider::getLastOrderUpdate();
|
||||
$dbOrderUpdate = OrderTable::GetList([
|
||||
'order' => ['ID' => 'ASC'],
|
||||
'filter' => ['>DATE_UPDATE' => DateTime::createFromTimestamp(strtotime($dateOfLastUpdate))],
|
||||
'limit' => $pSize,
|
||||
'select' => ['ID'],
|
||||
]);
|
||||
|
||||
while ($arOrderUpdate = $dbOrderUpdate->fetch()) {
|
||||
$orderUpdateIds[] = $arOrderUpdate['ID'];
|
||||
}
|
||||
}
|
||||
|
||||
$orderIds = array_unique(array_merge($orderIds, $orderUpdateIds));
|
||||
|
||||
if (count($orderIds) <= 0) {
|
||||
return false;
|
||||
}
|
||||
|
@ -588,6 +607,7 @@ class RetailCrmOrder
|
|||
];
|
||||
|
||||
$recOrders = [];
|
||||
$maxUpdateDate = $dateOfLastUpdate;
|
||||
|
||||
foreach ($orderIds as $orderId) {
|
||||
$bitrixOrder = Order::load($orderId);
|
||||
|
@ -598,6 +618,8 @@ class RetailCrmOrder
|
|||
|
||||
$arCustomer = [];
|
||||
$arCustomerCorporate = [];
|
||||
$methodApi = 'ordersCreate';
|
||||
|
||||
$order = self::orderObjToArr($bitrixOrder);
|
||||
$site = self::getCrmShopCodeByLid($order['LID'], $arParams['optionsSitesList']);
|
||||
|
||||
|
@ -615,14 +637,30 @@ class RetailCrmOrder
|
|||
continue;
|
||||
}
|
||||
|
||||
self::createCustomerForOrder($api, $arCustomer, $arCustomerCorporate,$arParams, $order, $site);
|
||||
if((int)$uploadMethod === 0) {
|
||||
$orderCrm = RCrmActions::apiMethod($api, 'ordersGet', __METHOD__, $orderId, $site);
|
||||
|
||||
if (isset($orderCrm['order'])) {
|
||||
$methodApi = 'ordersEdit';
|
||||
$arParams['crmOrder'] = $orderCrm['order'];
|
||||
}
|
||||
}
|
||||
|
||||
if (isset($order['RESPONSIBLE_ID']) && !empty($order['RESPONSIBLE_ID'])) {
|
||||
$managerService = ManagerService::getInstance();
|
||||
$arParams['managerId'] = $managerService->getManagerCrmId((int) $order['RESPONSIBLE_ID']);
|
||||
}
|
||||
|
||||
$arOrders = self::orderSend($order, $api, $arParams, false, $site,'ordersCreate');
|
||||
if ($methodApi === 'ordersEdit') {
|
||||
$updateDate = $bitrixOrder->getField('DATE_UPDATE')->format("Y-m-d H:i:s");
|
||||
$maxUpdateDate = $updateDate > $maxUpdateDate ? $updateDate : $maxUpdateDate;
|
||||
|
||||
self::orderSend($order, $api, $arParams, true, $site);
|
||||
continue;
|
||||
}
|
||||
|
||||
self::createCustomerForOrder($api, $arCustomer, $arCustomerCorporate,$arParams, $order, $site);
|
||||
$arOrders = self::orderSend($order, $api, $arParams, false, $site, $methodApi);
|
||||
|
||||
if (!$arCustomer || !$arOrders) {
|
||||
continue;
|
||||
|
@ -644,6 +682,8 @@ class RetailCrmOrder
|
|||
$recOrders[] = $orderId;
|
||||
}
|
||||
|
||||
RetailcrmConfigProvider::setLastOrderUpdate($maxUpdateDate);
|
||||
|
||||
if (count($ordersPack) > 0) {
|
||||
if (false === RetailCrmOrder::uploadCustomersList($resCustomers, $api, $arParams['optionsSitesList'])) {
|
||||
return false;
|
||||
|
@ -995,7 +1035,9 @@ class RetailCrmOrder
|
|||
}
|
||||
if ($delivery['PARENT_ID']) {
|
||||
$service = explode(':', $delivery['CODE']);
|
||||
$shipment = ['id' => $delivery['PARENT_ID'], 'service' => $service[1]];
|
||||
$shipment = $delivery['CLASS_NAME'] === '\Sale\Handlers\Delivery\RussianpostProfile'
|
||||
? ['id' => $delivery['PARENT_ID'], 'service' => 'bitrix-' . $delivery['ID']]
|
||||
: ['id' => $delivery['PARENT_ID'], 'service' => $service[1]];
|
||||
} else {
|
||||
$shipment = ['id' => $delivery['ID']];
|
||||
}
|
||||
|
|
|
@ -1 +1 @@
|
|||
- Исправлена подписка на событие сохранения заказа при обновлении
|
||||
- Исправлена передача габаритов при выгрузке заказов по агенту
|
|
@ -38,7 +38,8 @@ if (file_exists($_SERVER['DOCUMENT_ROOT'] . '/bitrix/php_interface/retailcrm/exp
|
|||
}
|
||||
}
|
||||
|
||||
$settingService = SettingsService::getInstance([], '');
|
||||
global $PROFILE_ID;
|
||||
$settingService = SettingsService::getInstance([], '', $PROFILE_ID ?? $profile_id);
|
||||
$iblockPropertySku = [];
|
||||
$iblockPropertySkuHl = [];
|
||||
$iblockPropertyUnitSku = [];
|
||||
|
@ -46,7 +47,7 @@ if (file_exists($_SERVER['DOCUMENT_ROOT'] . '/bitrix/php_interface/retailcrm/exp
|
|||
$iblockPropertyProductHl = [];
|
||||
$iblockPropertyUnitProduct = [];
|
||||
|
||||
foreach (array_keys($settingService->actrualPropList) as $prop) {
|
||||
foreach (array_keys($settingService->actualPropList) as $prop) {
|
||||
$skuUnitProps = ('iblockPropertyUnitSku_' . $prop);
|
||||
$skuUnitProps = $$skuUnitProps;
|
||||
|
||||
|
|
|
@ -26,7 +26,8 @@ CModule::IncludeModule('intaro.retailcrm');
|
|||
//TODO заменить вызов на сервис-локатор, когда он приедет
|
||||
$settingsService = SettingsService::getInstance(
|
||||
$arOldSetupVars ?? [],
|
||||
$ACTION
|
||||
$ACTION,
|
||||
$PROFILE_ID
|
||||
);
|
||||
|
||||
$isSetupModulePage = $settingsService->isSetupModulePage();
|
||||
|
@ -110,7 +111,7 @@ if ($STEP === 1) {
|
|||
}
|
||||
</style>
|
||||
|
||||
<form method="post" action="<?=$APPLICATION->GetCurPage()?>">
|
||||
<form method="post" id="submit-form" action="<?=$APPLICATION->GetCurPage()?>">
|
||||
<?php
|
||||
if ($ACTION === 'EXPORT_EDIT' || $ACTION === 'EXPORT_COPY') {
|
||||
?>
|
||||
|
@ -190,8 +191,7 @@ if ($STEP === 1) {
|
|||
<tbody>
|
||||
|
||||
<?php
|
||||
foreach ($settingsService->actrualPropList as $propertyKey => $property) {
|
||||
$productSelected = false; ?>
|
||||
foreach ($settingsService->defaultPropList as $propertyKey => $property) { ?>
|
||||
|
||||
<tr class="adm-list-table-row">
|
||||
<td class="adm-list-table-cell">
|
||||
|
@ -215,14 +215,14 @@ if ($STEP === 1) {
|
|||
if ($keyField === $propertyKey) { ?>
|
||||
<option value="<?=$field['CODE']?>"
|
||||
<?php
|
||||
$productSelected = $settingsService->isOptionSelected(
|
||||
$isSelected = $settingsService->isOptionSelected(
|
||||
$field,
|
||||
$arIBlock['OLD_PROPERTY_PRODUCT_SELECT'],
|
||||
$propertyKey
|
||||
);
|
||||
?>
|
||||
|
||||
<?= $productSelected ? ' selected' : ''?>
|
||||
<?= $isSelected ? ' selected' : ''?>
|
||||
>
|
||||
<?=$field['name']?>
|
||||
</option>
|
||||
|
@ -241,7 +241,7 @@ if ($STEP === 1) {
|
|||
<?php
|
||||
echo $settingsService->getOptionClass($prop, true);
|
||||
|
||||
$productSelected = $settingsService->isOptionSelected(
|
||||
$isSelected = $settingsService->isOptionSelected(
|
||||
$prop,
|
||||
$arIBlock['OLD_PROPERTY_PRODUCT_SELECT'],
|
||||
$propertyKey
|
||||
|
@ -251,7 +251,7 @@ if ($STEP === 1) {
|
|||
= $settingsService->getHlTableName($prop)
|
||||
?? $productHlTableName;
|
||||
|
||||
echo $productSelected ? ' selected' : '';
|
||||
echo $isSelected ? ' selected' : '';
|
||||
?>
|
||||
>
|
||||
<?=$prop['NAME']?>
|
||||
|
@ -376,19 +376,18 @@ if ($STEP === 1) {
|
|||
<option value="<?=$prop['CODE']?>"
|
||||
<?php
|
||||
echo $settingsService->getOptionClass($prop, false);
|
||||
if (!$productSelected) {
|
||||
$isSelected = $settingsService->isOptionSelected(
|
||||
$prop,
|
||||
$arIBlock['OLD_PROPERTY_SKU_SELECT'],
|
||||
$propertyKey
|
||||
);
|
||||
|
||||
$skuHlTableName
|
||||
= $settingsService->getHlTableName($prop)
|
||||
?? $skuHlTableName;
|
||||
$isSelected = $settingsService->isOptionSelected(
|
||||
$prop,
|
||||
$arIBlock['OLD_PROPERTY_SKU_SELECT'],
|
||||
$propertyKey
|
||||
);
|
||||
|
||||
echo $isSelected ? ' selected' : '';
|
||||
}
|
||||
$skuHlTableName
|
||||
= $settingsService->getHlTableName($prop)
|
||||
?? $skuHlTableName;
|
||||
|
||||
echo $isSelected ? ' selected' : '';
|
||||
?>
|
||||
>
|
||||
<?=$prop['NAME']?>
|
||||
|
@ -473,9 +472,97 @@ if ($STEP === 1) {
|
|||
</tr>
|
||||
<?php
|
||||
} ?>
|
||||
|
||||
<?php
|
||||
$catalogId = $arIBlock['ID'];
|
||||
$catalogCustomProps = $settingsService->customPropList[$catalogId];
|
||||
|
||||
if (!empty($catalogCustomProps)) {
|
||||
foreach ($catalogCustomProps as $catalogCustomPropCode => $catalogCustomPropValue) { ?>
|
||||
<tr class="adm-list-table-row custom-property-row">
|
||||
<td class="adm-list-table-cell custom-property-title">
|
||||
<?=htmlspecialcharsex($catalogCustomPropValue)?>
|
||||
</td>
|
||||
<td class="adm-list-table-cell">
|
||||
<select
|
||||
name="iblockPropertyProduct_<?=$catalogCustomPropCode?>[<?= $catalogId ?>]"
|
||||
id="iblockPropertyProduct_<?=$catalogCustomPropCode . $catalogId ?>"
|
||||
class="property-export"
|
||||
data-type="<?= $catalogCustomPropCode ?>"
|
||||
onchange="propertyChange(this)"
|
||||
style="width: 200px">
|
||||
|
||||
<option value=""></option>
|
||||
<?php foreach ($arIBlock['PROPERTIES_PRODUCT'] as $prop) {
|
||||
$productHlTableName = ''; ?>
|
||||
<option value="<?=$prop['CODE']?>"
|
||||
<?php
|
||||
echo $settingsService->getOptionClass($prop, true);
|
||||
|
||||
$isSelected = $settingsService->isOptionSelected(
|
||||
$prop,
|
||||
$arIBlock['OLD_PROPERTY_PRODUCT_SELECT'],
|
||||
$catalogCustomPropCode
|
||||
);
|
||||
|
||||
$productHlTableName
|
||||
= $settingsService->getHlTableName($prop)
|
||||
?? $productHlTableName;
|
||||
|
||||
echo $isSelected ? ' selected' : '';
|
||||
?>
|
||||
>
|
||||
<?=$prop['NAME']?>
|
||||
</option>
|
||||
<?php } ?>
|
||||
</select>
|
||||
</td>
|
||||
|
||||
<?php if ($arIBlock['PROPERTIES_SKU'] !== null) { ?>
|
||||
<td class="adm-list-table-cell">
|
||||
<select
|
||||
name="iblockPropertySku_<?=$catalogCustomPropCode?>[<?= $catalogId ?>]"
|
||||
id="iblockPropertySku_<?=$catalogCustomPropCode . $catalogId ?>"
|
||||
class="property-export"
|
||||
data-type="<?= $catalogCustomPropCode ?>"
|
||||
onchange="propertyChange(this)"
|
||||
style="width: 200px">
|
||||
|
||||
<option value=""></option>
|
||||
<?php foreach ($arIBlock['PROPERTIES_SKU'] as $prop) {
|
||||
$skuHlTableName = ''; ?>
|
||||
<option value="<?=$prop['CODE']?>"
|
||||
<?php
|
||||
echo $settingsService->getOptionClass($prop, false);
|
||||
|
||||
$isSelected = $settingsService->isOptionSelected(
|
||||
$prop,
|
||||
$arIBlock['OLD_PROPERTY_SKU_SELECT'],
|
||||
$catalogCustomPropCode
|
||||
);
|
||||
|
||||
$skuHlTableName
|
||||
= $settingsService->getHlTableName($prop)
|
||||
?? $skuHlTableName;
|
||||
|
||||
echo $isSelected ? ' selected' : '';
|
||||
?>
|
||||
>
|
||||
<?=$prop['NAME']?>
|
||||
</option>
|
||||
|
||||
<?php } ?>
|
||||
</select>
|
||||
<button id="delete-custom-row" class="adm-btn-save" type="button" style="margin-left: 10px"><?= GetMessage('DELETE_PROPERTY');?></button>
|
||||
</td>
|
||||
<?php } ?>
|
||||
</tr>
|
||||
<?php }
|
||||
}
|
||||
?>
|
||||
</tbody>
|
||||
</table>
|
||||
<br>
|
||||
<button class="adm-btn-save add-custom-row" type="button" style="margin-top: 20px;"><?= GetMessage('ADD_PROPERTY');?></button>
|
||||
<br>
|
||||
</div>
|
||||
</div>
|
||||
|
@ -484,6 +571,20 @@ if ($STEP === 1) {
|
|||
</div>
|
||||
<input type="hidden" name="count_checked" id="count_checked" value="<?=$intCountChecked?>">
|
||||
<br>
|
||||
<template id="custom-property-template-row">
|
||||
<tr class="adm-list-table-row custom-property-row">
|
||||
<td class="adm-list-table-cell">
|
||||
<input type="text" title="Название нового свойства" name="custom-property-title" style="width: 200px">
|
||||
</td>
|
||||
<td class="adm-list-table-cell">
|
||||
<select name="iblockPropertyProduct_" id="iblockPropertyProduct_" class="property-export" onchange="propertyChange(this)" style="width: 200px"></select>
|
||||
</td>
|
||||
<td class="adm-list-table-cell">
|
||||
<select name="iblockPropertySku_" id="iiblockPropertySku_" class="property-export" onchange="propertyChange(this)" style="width: 200px"></select>
|
||||
<button id="delete-new-custom-row" class="adm-btn-save" type="button" style="margin-left: 10px"><?= GetMessage('DELETE_PROPERTY');?></button>
|
||||
</td>
|
||||
</tr>
|
||||
</template>
|
||||
<h3><?=GetMessage('SETTINGS_EXPORT')?></h3>
|
||||
<span class="text"><?=GetMessage('FILENAME')?><br><br></span>
|
||||
<input type="text" name="SETUP_FILE_NAME" value="<?=htmlspecialcharsbx(strlen($SETUP_FILE_NAME) > 0 ?
|
||||
|
@ -566,7 +667,7 @@ if ($STEP === 1) {
|
|||
<input type="hidden" name="STEP" value="<?=$STEP + 1?>">
|
||||
<input type="hidden" name="SETUP_FIELDS_LIST" value="<?=
|
||||
$settingsService->getSetupFieldsString(
|
||||
array_keys($settingsService->actrualPropList) ?? [],
|
||||
array_keys($settingsService->actualPropList) ?? [],
|
||||
$hlblockModule === true,
|
||||
$hlBlockList ?? []
|
||||
)
|
||||
|
@ -577,8 +678,109 @@ if ($STEP === 1) {
|
|||
</form>
|
||||
|
||||
<?php CJSCore::Init(['jquery']);?>
|
||||
|
||||
<?php CUtil::InitJSCore(['intaro_custom_props']); ?>
|
||||
<script type="text/javascript">
|
||||
BX.ready(function() {
|
||||
if (typeof createCustomPropsRaw !== 'function') {
|
||||
$('.add-custom-row').hide();
|
||||
}
|
||||
});
|
||||
|
||||
$('.add-custom-row').click(function () {
|
||||
if (typeof createCustomPropsRaw === 'function') {
|
||||
createCustomPropsRaw($(this));
|
||||
}
|
||||
});
|
||||
|
||||
$(document).on('click', '#delete-new-custom-row', function () {
|
||||
if (typeof deleteCustomPropRow === 'function') {
|
||||
deleteCustomPropRow($(this));
|
||||
}
|
||||
});
|
||||
|
||||
$(document).on('click', '#delete-custom-row', function () {
|
||||
let buttonElem = $(this);
|
||||
|
||||
if (typeof addCustomPropToDelete === 'function' && typeof deleteCustomPropRow === 'function') {
|
||||
addCustomPropToDelete(buttonElem);
|
||||
deleteCustomPropRow(buttonElem);
|
||||
}
|
||||
});
|
||||
|
||||
$(document).on('blur', 'input[name="custom-property-title"]', function () {
|
||||
|
||||
if (typeof getUniquePropertyCode === 'function' && typeof addCustomPropCodeToSelectAttributes === 'function') {
|
||||
let inputElem = $(this);
|
||||
let newPropertyTitle = inputElem.val();
|
||||
|
||||
if (!newPropertyTitle) {
|
||||
return;
|
||||
}
|
||||
|
||||
let newPropertyCode = getUniquePropertyCode(newPropertyTitle);
|
||||
addCustomPropCodeToSelectAttributes(newPropertyCode, inputElem);
|
||||
}
|
||||
});
|
||||
|
||||
$('#submit-form').submit(function (formEvent) {
|
||||
if (typeof setCustomProperties !== "function") {
|
||||
return;
|
||||
}
|
||||
|
||||
formEvent.preventDefault();
|
||||
let savePromise = null;
|
||||
let deletePromise = null;
|
||||
let formElem = formEvent.currentTarget;
|
||||
let profileId = $($('input[name="PROFILE_ID"]')).val();
|
||||
|
||||
setCustomProperties();
|
||||
|
||||
if (Object.keys(customProps).length > 0) {
|
||||
savePromise = BX.ajax.runAction('intaro:retailcrm.api.customexportprops.save', {
|
||||
json: {
|
||||
properties: customProps,
|
||||
profileId: profileId
|
||||
},
|
||||
}).then(addParamsToSetupFieldsList());
|
||||
}
|
||||
|
||||
if (Object.keys(customPropsToDelete).length > 0) {
|
||||
deletePromise = BX.ajax.runAction('intaro:retailcrm.api.customexportprops.delete', {
|
||||
json: {
|
||||
properties: customPropsToDelete,
|
||||
profileId: profileId
|
||||
},
|
||||
}).then(deleteParamsFromSetupFieldsList());
|
||||
}
|
||||
|
||||
const promises = [savePromise, deletePromise].filter(Boolean);
|
||||
|
||||
if (promises.length > 0) {
|
||||
Promise.all(promises)
|
||||
.finally(() => {
|
||||
formElem.submit();
|
||||
});
|
||||
} else {
|
||||
formElem.submit();
|
||||
}
|
||||
});
|
||||
|
||||
const setupFieldsListElement = $('input[name="SETUP_FIELDS_LIST"]');
|
||||
let customProps = {};
|
||||
let customPropsToDelete = {};
|
||||
const setupFieldsParamsToFill = [
|
||||
'iblockPropertySku_',
|
||||
'iblockPropertyUnitSku_',
|
||||
'iblockPropertyProduct_',
|
||||
'iblockPropertyUnitProduct_',
|
||||
'highloadblockb_hlsys_marking_code_group_',
|
||||
'highloadblock_productb_hlsys_marking_code_group_',
|
||||
'highloadblockeshop_color_reference_',
|
||||
'highloadblock_producteshop_color_reference_',
|
||||
'highloadblockeshop_brand_reference_',
|
||||
'highloadblock_producteshop_brand_reference_'
|
||||
];
|
||||
|
||||
function checkLoadStatus(object)
|
||||
{
|
||||
if (object.checked) {
|
||||
|
|
|
@ -50,6 +50,10 @@ $arJsConfig = [
|
|||
'js' => '/bitrix/js/intaro/sms.js',
|
||||
'rel' => [],
|
||||
],
|
||||
'intaro_custom_props' => [
|
||||
'js' => '/bitrix/js/intaro/export/custom-props-export.js',
|
||||
'rel' => [],
|
||||
],
|
||||
];
|
||||
|
||||
foreach ($arJsConfig as $ext => $arExt) {
|
||||
|
|
|
@ -0,0 +1,228 @@
|
|||
|
||||
function deleteCustomPropRow(deleteButton)
|
||||
{
|
||||
deleteButton.closest('tr').remove();
|
||||
}
|
||||
|
||||
function addCustomPropToDelete(deleteButton)
|
||||
{
|
||||
let deletedPropTitle = deleteButton.closest('td').siblings().filter('.custom-property-title').text().trim();
|
||||
let deletedPropCode = deleteButton.siblings().filter('select').first().data('type');
|
||||
let customPropCatalogId = deleteButton.closest('.iblockExportTable').data('type');
|
||||
|
||||
let values = {
|
||||
'code': deletedPropCode,
|
||||
'title': deletedPropTitle,
|
||||
};
|
||||
|
||||
if (customPropsToDelete.hasOwnProperty(customPropCatalogId)) {
|
||||
customPropsToDelete[customPropCatalogId].push(values);
|
||||
} else {
|
||||
customPropsToDelete[customPropCatalogId] = [values];
|
||||
}
|
||||
}
|
||||
|
||||
function addCustomPropCodeToSelectAttributes(customPropCode, customPropTitleElem)
|
||||
{
|
||||
let selectElements = customPropTitleElem.closest('.custom-property-row').find('td select');
|
||||
let catalogId = customPropTitleElem.closest('.iblockExportTable').data('type');
|
||||
|
||||
selectElements.each(function (index, element) {
|
||||
let selectElem = $(element);
|
||||
let newSelectIdValue = selectElem.attr('id').match(/^[^_]*_/)[0] + customPropCode + catalogId;
|
||||
let newSelectNameValue = selectElem.attr('name').match(/^[^_]*_/)[0] + customPropCode + `[${catalogId}]`;
|
||||
|
||||
selectElem.attr('id', newSelectIdValue);
|
||||
selectElem.attr('name', newSelectNameValue);
|
||||
selectElem.data('type', customPropCode);
|
||||
triggerSelectChange(selectElem);
|
||||
});
|
||||
}
|
||||
|
||||
function triggerSelectChange(selectElem)
|
||||
{
|
||||
if (selectElem.val().length > 0) {
|
||||
selectElem.trigger('change', [self]);
|
||||
}
|
||||
}
|
||||
|
||||
function setCustomProperties()
|
||||
{
|
||||
let customPropertiesRows = $('.custom-property-row');
|
||||
|
||||
if (customPropertiesRows.length === 0) {
|
||||
return;
|
||||
}
|
||||
|
||||
let customPropertyCatalogId;
|
||||
let customPropertyTitle = '';
|
||||
let customPropertyCode = '';
|
||||
let productPropertyMatch = '';
|
||||
let offerPropertyMatch = '';
|
||||
|
||||
let catalogIds = [];
|
||||
customPropertiesRows.each(function (index, propertyRow) {
|
||||
let propertyRowObj = $(propertyRow);
|
||||
customPropertyCatalogId = propertyRowObj.closest('.iblockExportTable').data('type');
|
||||
customPropertyTitle = propertyRowObj.find('input[name="custom-property-title"]').val();
|
||||
|
||||
if (!customPropertyTitle) {
|
||||
return true;
|
||||
}
|
||||
|
||||
customPropertyCode = getUniquePropertyCode(customPropertyTitle);
|
||||
productPropertyMatch = propertyRowObj.find('select[name=custom-product-property-select]').val();
|
||||
offerPropertyMatch = propertyRowObj.find('select[name=custom-offer-property-select]').val();
|
||||
|
||||
let values = {
|
||||
'title': customPropertyTitle,
|
||||
'code': customPropertyCode,
|
||||
'productProperty': productPropertyMatch,
|
||||
'offerProperty': offerPropertyMatch
|
||||
};
|
||||
|
||||
if (catalogIds.indexOf(customPropertyCatalogId) === -1) {
|
||||
customProps[customPropertyCatalogId] = [values];
|
||||
} else {
|
||||
customProps[customPropertyCatalogId].push(values);
|
||||
}
|
||||
|
||||
catalogIds.push(customPropertyCatalogId);
|
||||
});
|
||||
}
|
||||
|
||||
function getUniquePropertyCode(customPropertyTitle)
|
||||
{
|
||||
let uniqueValue = transliterate(customPropertyTitle).replace(/ /g, '_');
|
||||
let counter = 0;
|
||||
const setupFieldsListValues = setupFieldsListElement.val().split(',');
|
||||
|
||||
while (setupFieldsListValues.includes(uniqueValue)) {
|
||||
uniqueValue = `${customPropertyTitle}${++counter}`;
|
||||
}
|
||||
|
||||
return uniqueValue;
|
||||
}
|
||||
|
||||
function addParamsToSetupFieldsList()
|
||||
{
|
||||
let newParams = '';
|
||||
|
||||
if (Object.keys(customProps).length === 0) {
|
||||
return;
|
||||
}
|
||||
|
||||
for (let propertiesByCatalogId of Object.values(customProps)) {
|
||||
propertiesByCatalogId.forEach(function (values) {
|
||||
setupFieldsParamsToFill.forEach(function (param) {
|
||||
newParams += ',' + param + values.code;
|
||||
});
|
||||
});
|
||||
}
|
||||
|
||||
let newValue = setupFieldsListElement.val() + newParams;
|
||||
setupFieldsListElement.val(newValue);
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
function deleteParamsFromSetupFieldsList()
|
||||
{
|
||||
let setupFields = setupFieldsListElement.val();
|
||||
|
||||
if (Object.keys(customPropsToDelete).length === 0) {
|
||||
return;
|
||||
}
|
||||
|
||||
for (let propsByCatalogId of Object.values(customPropsToDelete)) {
|
||||
propsByCatalogId.forEach(function (propValues) {
|
||||
setupFieldsParamsToFill.forEach(function (param) {
|
||||
let paramToDelete = ',' + param + propValues.code;
|
||||
setupFields = setupFields.replace(paramToDelete, '');
|
||||
});
|
||||
});
|
||||
}
|
||||
|
||||
setupFieldsListElement.val(setupFields);
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
function createCustomPropsRaw(addRowButton)
|
||||
{
|
||||
let templateRow = $($('#custom-property-template-row').html());
|
||||
let templateSelectElements = templateRow.find('select');
|
||||
|
||||
let prevTableRow = $(addRowButton).prev('table').find('tbody tr:last-child');
|
||||
let lastRawSelectElements = prevTableRow.find('td select');
|
||||
|
||||
lastRawSelectElements.each(function (index, element) {
|
||||
let selectElement = $(element);
|
||||
let templateSelectElement = templateSelectElements[index];
|
||||
fillTemplateSelect(selectElement, templateSelectElement);
|
||||
prevTableRow.after(templateRow);
|
||||
});
|
||||
}
|
||||
|
||||
function fillTemplateSelect(sourceSelectElement, templateSelectElement)
|
||||
{
|
||||
let selectOptions = sourceSelectElement.find('option');
|
||||
|
||||
selectOptions.each(function (index, element) {
|
||||
let optionElem = $(element);
|
||||
let value = $(optionElem).val();
|
||||
let text = $(optionElem).text();
|
||||
|
||||
$('<option>', { value: value, text: text }).appendTo(templateSelectElement);
|
||||
});
|
||||
}
|
||||
|
||||
function transliterate(titleToTransliterate)
|
||||
{
|
||||
const hasCyrillicChars = /[\u0400-\u04FF]/.test(titleToTransliterate);
|
||||
|
||||
if (!hasCyrillicChars) {
|
||||
return titleToTransliterate;
|
||||
}
|
||||
|
||||
translitedText = '';
|
||||
for (var i = 0; i < titleToTransliterate.length; i++) {
|
||||
switch (titleToTransliterate[i]) {
|
||||
case 'а': case 'А': translitedText += 'a'; break;
|
||||
case 'б': case 'Б': translitedText += 'b'; break;
|
||||
case 'в': case 'В': translitedText += 'v'; break;
|
||||
case 'г': case 'Г': translitedText += 'g'; break;
|
||||
case 'д': case 'Д': translitedText += 'd'; break;
|
||||
case 'е': case 'Е': translitedText += 'e'; break;
|
||||
case 'ё': case 'Ё': translitedText += 'yo'; break;
|
||||
case 'ж': case 'Ж': translitedText += 'zh'; break;
|
||||
case 'з': case 'З': translitedText += 'z'; break;
|
||||
case 'и': case 'И': translitedText += 'i'; break;
|
||||
case 'й': case 'Й': translitedText += 'y'; break;
|
||||
case 'к': case 'К': translitedText += 'k'; break;
|
||||
case 'л': case 'Л': translitedText += 'l'; break;
|
||||
case 'м': case 'М': translitedText += 'm'; break;
|
||||
case 'н': case 'Н': translitedText += 'n'; break;
|
||||
case 'о': case 'О': translitedText += 'o'; break;
|
||||
case 'п': case 'П': translitedText += 'p'; break;
|
||||
case 'р': case 'Р': translitedText += 'r'; break;
|
||||
case 'с': case 'С': translitedText += 's'; break;
|
||||
case 'т': case 'Т': translitedText += 't'; break;
|
||||
case 'у': case 'У': translitedText += 'u'; break;
|
||||
case 'ф': case 'Ф': translitedText += 'f'; break;
|
||||
case 'х': case 'Х': translitedText += 'h'; break;
|
||||
case 'ц': case 'Ц': translitedText += 'c'; break;
|
||||
case 'ч': case 'Ч': translitedText += 'ch'; break;
|
||||
case 'ш': case 'Ш': translitedText += 'sh'; break;
|
||||
case 'щ': case 'Щ': translitedText += 'sch'; break;
|
||||
case 'ъ': case 'Ъ': translitedText += ''; break;
|
||||
case 'ы': case 'Ы': translitedText += 'y'; break;
|
||||
case 'ь': case 'Ь': translitedText += ''; break;
|
||||
case 'э': case 'Э': translitedText += 'e'; break;
|
||||
case 'ю': case 'Ю': translitedText += 'yu'; break;
|
||||
case 'я': case 'Я': translitedText += 'ya'; break;
|
||||
default: translitedText += titleToTransliterate[i]; break;
|
||||
}
|
||||
}
|
||||
return translitedText;
|
||||
}
|
|
@ -26,11 +26,12 @@ use Intaro\RetailCrm\Component\Advanced\LoyaltyInstaller;
|
|||
Loader::IncludeModule('highloadblock');
|
||||
|
||||
IncludeModuleLangFile(__FILE__);
|
||||
|
||||
if (class_exists('intaro_retailcrm')) {
|
||||
return false;
|
||||
}
|
||||
|
||||
include (__DIR__ . '/../lib/component/advanced/installertrait.php');
|
||||
require_once(__DIR__ . '/../lib/component/advanced/installertrait.php');
|
||||
|
||||
class intaro_retailcrm extends CModule
|
||||
{
|
||||
|
@ -51,11 +52,13 @@ class intaro_retailcrm extends CModule
|
|||
|
||||
public function __construct()
|
||||
{
|
||||
$arModuleVersion = [];
|
||||
$path = str_replace("\\", '/', __FILE__);
|
||||
$path = substr($path, 0, strlen($path) - strlen('/index.php'));
|
||||
$arModuleVersion = [];
|
||||
$path = str_replace("\\", '/', __FILE__);
|
||||
$path = substr($path, 0, strlen($path) - strlen('/index.php'));
|
||||
$this->INSTALL_PATH = $path;
|
||||
|
||||
include($path . '/version.php');
|
||||
|
||||
$this->MODULE_VERSION = $arModuleVersion['VERSION'];
|
||||
$this->MODULE_VERSION_DATE = $arModuleVersion['VERSION_DATE'];
|
||||
$this->MODULE_NAME = GetMessage('RETAIL_MODULE_NAME');
|
||||
|
@ -85,6 +88,18 @@ class intaro_retailcrm extends CModule
|
|||
*/
|
||||
function DoInstall()
|
||||
{
|
||||
try {
|
||||
require_once('install_deps.php');
|
||||
} catch (Error $exception) {
|
||||
RCrmActions::eventLog(
|
||||
'RetailCRM module install error',
|
||||
'intaro.retailcrm',
|
||||
$exception->getCode() . ': ' . $exception->getMessage()
|
||||
);
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
global $APPLICATION, $step, $arResult;
|
||||
|
||||
if (!in_array('curl', get_loaded_extensions(), true)) {
|
||||
|
@ -110,57 +125,6 @@ class intaro_retailcrm extends CModule
|
|||
return false;
|
||||
}
|
||||
|
||||
include($this->INSTALL_PATH . '/../lib/component/apiclient/traits/baseclienttrait.php');
|
||||
include($this->INSTALL_PATH . '/../lib/component/apiclient/traits/customerstrait.php');
|
||||
include($this->INSTALL_PATH . '/../lib/component/apiclient/traits/customerscorporatetrait.php');
|
||||
include($this->INSTALL_PATH . '/../lib/component/apiclient/traits/loyaltytrait.php');
|
||||
include($this->INSTALL_PATH . '/../lib/component/apiclient/traits/ordertrait.php');
|
||||
include($this->INSTALL_PATH . '/../lib/component/apiclient/traits/carttrait.php');
|
||||
include($this->INSTALL_PATH . '/../classes/general/Http/Client.php');
|
||||
include($this->INSTALL_PATH . '/../classes/general/Response/ApiResponse.php');
|
||||
include($this->INSTALL_PATH . '/../classes/general/RCrmActions.php');
|
||||
include($this->INSTALL_PATH . '/../classes/general/user/RetailCrmUser.php');
|
||||
include($this->INSTALL_PATH . '/../classes/general/events/RetailCrmEvent.php');
|
||||
require_once $this->INSTALL_PATH . '/../classes/general/RetailcrmConfigProvider.php';
|
||||
include($this->INSTALL_PATH . '/../lib/model/bitrix/xml/offerparam.php');
|
||||
include($this->INSTALL_PATH . '/../lib/icml/settingsservice.php');
|
||||
include($this->INSTALL_PATH . '/../lib/component/agent.php');
|
||||
include($this->INSTALL_PATH . '/../lib/model/bitrix/xml/selectparams.php');
|
||||
include($this->INSTALL_PATH . '/../lib/model/bitrix/xml/unit.php');
|
||||
include($this->INSTALL_PATH . '/../lib/model/bitrix/xml/xmlcategory.php');
|
||||
include($this->INSTALL_PATH . '/../lib/model/bitrix/xml/xmldata.php');
|
||||
include($this->INSTALL_PATH . '/../lib/model/bitrix/xml/xmloffer.php');
|
||||
include($this->INSTALL_PATH . '/../lib/model/bitrix/xml/xmlsetup.php');
|
||||
include($this->INSTALL_PATH . '/../lib/model/bitrix/xml/xmlsetupprops.php');
|
||||
include($this->INSTALL_PATH . '/../lib/model/bitrix/xml/xmlsetuppropscategories.php');
|
||||
include($this->INSTALL_PATH . '/../lib/icml/icmldirector.php');
|
||||
include($this->INSTALL_PATH . '/../lib/icml/icmlwriter.php');
|
||||
include($this->INSTALL_PATH . '/../lib/icml/queryparamsmolder.php');
|
||||
include($this->INSTALL_PATH . '/../lib/icml/xmlcategorydirector.php');
|
||||
include($this->INSTALL_PATH . '/../lib/icml/xmlcategoryfactory.php');
|
||||
include($this->INSTALL_PATH . '/../lib/icml/xmlofferdirector.php');
|
||||
include($this->INSTALL_PATH . '/../lib/icml/xmlofferbuilder.php');
|
||||
include($this->INSTALL_PATH . '/../lib/icml/utils/icmlutils.php');
|
||||
include($this->INSTALL_PATH . '/../lib/repository/catalogrepository.php');
|
||||
include($this->INSTALL_PATH . '/../lib/repository/filerepository.php');
|
||||
include($this->INSTALL_PATH . '/../lib/repository/hlrepository.php');
|
||||
include($this->INSTALL_PATH . '/../lib/repository/measurerepository.php');
|
||||
include($this->INSTALL_PATH . '/../lib/repository/siterepository.php');
|
||||
include($this->INSTALL_PATH . '/../lib/service/hl.php');
|
||||
include($this->INSTALL_PATH . '/../lib/model/bitrix/orm/catalogiblockinfo.php');
|
||||
include($this->INSTALL_PATH . '/../lib/model/bitrix/orm/iblockcatalog.php');
|
||||
include($this->INSTALL_PATH . '/../classes/general/Exception/InvalidJsonException.php');
|
||||
include($this->INSTALL_PATH . '/../classes/general/Exception/CurlException.php');
|
||||
include($this->INSTALL_PATH . '/../classes/general/RestNormalizer.php');
|
||||
include($this->INSTALL_PATH . '/../classes/general/Logger.php');
|
||||
include($this->INSTALL_PATH . '/../classes/general/services/RetailCrmService.php');
|
||||
include($this->INSTALL_PATH . '/../lib/component/constants.php');
|
||||
$version = COption::GetOptionString($this->MODULE_ID, Constants::CRM_API_VERSION, 0);
|
||||
include($this->INSTALL_PATH . '/../classes/general/ApiClient_v5.php');
|
||||
include($this->INSTALL_PATH . '/../classes/general/order/RetailCrmOrder_v5.php');
|
||||
include($this->INSTALL_PATH . '/../classes/general/history/RetailCrmHistory_v5.php');
|
||||
include($this->INSTALL_PATH . '/../classes/general/cart/RetailCrmCart_v5.php');
|
||||
|
||||
$step = (int) $_REQUEST['step'];
|
||||
|
||||
if (file_exists($this->INSTALL_PATH . '/../classes/general/config/options.xml')) {
|
||||
|
@ -189,22 +153,6 @@ class intaro_retailcrm extends CModule
|
|||
}
|
||||
}
|
||||
|
||||
include($this->INSTALL_PATH . '/../lib/model/bitrix/abstractmodelproxy.php');
|
||||
include($this->INSTALL_PATH . '/../lib/model/bitrix/orderprops.php');
|
||||
include($this->INSTALL_PATH . '/../lib/model/bitrix/tomodule.php');
|
||||
include($this->INSTALL_PATH . '/../lib/repository/abstractrepository.php');
|
||||
include($this->INSTALL_PATH . '/../lib/repository/orderpropsrepository.php');
|
||||
include($this->INSTALL_PATH . '/../lib/repository/persontyperepository.php');
|
||||
include($this->INSTALL_PATH . '/../lib/repository/tomodulerepository.php');
|
||||
include($this->INSTALL_PATH . '/../lib/model/bitrix/orm/tomodule.php');
|
||||
include($this->INSTALL_PATH . '/../lib/model/bitrix/agreement.php');
|
||||
include($this->INSTALL_PATH . '/../lib/repository/agreementrepository.php');
|
||||
include($this->INSTALL_PATH . '/../lib/service/orderloyaltydataservice.php');
|
||||
include($this->INSTALL_PATH . '/../lib/service/currencyservice.php');
|
||||
include($this->INSTALL_PATH . '/../lib/component/factory/clientfactory.php');
|
||||
include($this->INSTALL_PATH . '/../lib/component/apiclient/clientadapter.php');
|
||||
include($this->INSTALL_PATH . '/../lib/component/advanced/loyaltyinstaller.php');
|
||||
|
||||
$this->installExport();
|
||||
$this->subscriptionSetup();
|
||||
|
||||
|
@ -1090,6 +1038,7 @@ class intaro_retailcrm extends CModule
|
|||
RegisterModuleDependences('sale', 'OnSalePaymentEntityDeleted', $this->MODULE_ID, 'RetailCrmEvent', 'paymentDelete');
|
||||
RegisterModuleDependences('main', 'OnAfterUserRegister', $this->MODULE_ID, 'RetailCrmEvent', 'OnAfterUserRegister');
|
||||
RegisterModuleDependences('main', 'OnAfterUserAdd', $this->MODULE_ID, 'RetailCrmEvent', 'OnAfterUserAdd');
|
||||
RegisterModuleDependences('sale', 'OnSaleOrderSaved', $this->MODULE_ID, 'RetailCrmEvent', 'orderSave', 99);
|
||||
|
||||
COption::SetOptionString($this->MODULE_ID, Constants::CRM_CATALOG_BASE_PRICE, htmlspecialchars(trim($_POST['price-types'])));
|
||||
COption::SetOptionString($this->MODULE_ID, Constants::CRM_INVENTORIES_UPLOAD, 'N');
|
||||
|
|
70
intaro.retailcrm/install/install_deps.php
Normal file
70
intaro.retailcrm/install/install_deps.php
Normal file
|
@ -0,0 +1,70 @@
|
|||
<?php
|
||||
|
||||
require_once($this->INSTALL_PATH . '/../lib/component/apiclient/traits/baseclienttrait.php');
|
||||
require_once($this->INSTALL_PATH . '/../lib/component/apiclient/traits/customerstrait.php');
|
||||
require_once($this->INSTALL_PATH . '/../lib/component/apiclient/traits/customerscorporatetrait.php');
|
||||
require_once($this->INSTALL_PATH . '/../lib/component/apiclient/traits/loyaltytrait.php');
|
||||
require_once($this->INSTALL_PATH . '/../lib/component/apiclient/traits/ordertrait.php');
|
||||
require_once($this->INSTALL_PATH . '/../lib/component/apiclient/traits/carttrait.php');
|
||||
require_once($this->INSTALL_PATH . '/../classes/general/Http/Client.php');
|
||||
require_once($this->INSTALL_PATH . '/../classes/general/Response/ApiResponse.php');
|
||||
require_once($this->INSTALL_PATH . '/../classes/general/RCrmActions.php');
|
||||
require_once($this->INSTALL_PATH . '/../classes/general/user/RetailCrmUser.php');
|
||||
require_once($this->INSTALL_PATH . '/../classes/general/events/RetailCrmEvent.php');
|
||||
require_once $this->INSTALL_PATH . '/../classes/general/RetailcrmConfigProvider.php';
|
||||
require_once($this->INSTALL_PATH . '/../lib/model/bitrix/xml/offerparam.php');
|
||||
require_once($this->INSTALL_PATH . '/../lib/icml/settingsservice.php');
|
||||
require_once($this->INSTALL_PATH . '/../lib/component/agent.php');
|
||||
require_once($this->INSTALL_PATH . '/../lib/model/bitrix/xml/selectparams.php');
|
||||
require_once($this->INSTALL_PATH . '/../lib/model/bitrix/xml/unit.php');
|
||||
require_once($this->INSTALL_PATH . '/../lib/model/bitrix/xml/xmlcategory.php');
|
||||
require_once($this->INSTALL_PATH . '/../lib/model/bitrix/xml/xmldata.php');
|
||||
require_once($this->INSTALL_PATH . '/../lib/model/bitrix/xml/xmloffer.php');
|
||||
require_once($this->INSTALL_PATH . '/../lib/model/bitrix/xml/xmlsetup.php');
|
||||
require_once($this->INSTALL_PATH . '/../lib/model/bitrix/xml/xmlsetupprops.php');
|
||||
require_once($this->INSTALL_PATH . '/../lib/model/bitrix/xml/xmlsetuppropscategories.php');
|
||||
require_once($this->INSTALL_PATH . '/../lib/icml/icmldirector.php');
|
||||
require_once($this->INSTALL_PATH . '/../lib/icml/icmlwriter.php');
|
||||
require_once($this->INSTALL_PATH . '/../lib/icml/queryparamsmolder.php');
|
||||
require_once($this->INSTALL_PATH . '/../lib/icml/xmlcategorydirector.php');
|
||||
require_once($this->INSTALL_PATH . '/../lib/icml/xmlcategoryfactory.php');
|
||||
require_once($this->INSTALL_PATH . '/../lib/icml/xmlofferdirector.php');
|
||||
require_once($this->INSTALL_PATH . '/../lib/icml/xmlofferbuilder.php');
|
||||
require_once($this->INSTALL_PATH . '/../lib/repository/catalogrepository.php');
|
||||
require_once($this->INSTALL_PATH . '/../lib/repository/filerepository.php');
|
||||
require_once($this->INSTALL_PATH . '/../lib/repository/hlrepository.php');
|
||||
require_once($this->INSTALL_PATH . '/../lib/repository/measurerepository.php');
|
||||
require_once($this->INSTALL_PATH . '/../lib/repository/siterepository.php');
|
||||
require_once($this->INSTALL_PATH . '/../lib/service/hl.php');
|
||||
require_once($this->INSTALL_PATH . '/../lib/model/bitrix/orm/catalogiblockinfo.php');
|
||||
require_once($this->INSTALL_PATH . '/../lib/model/bitrix/orm/iblockcatalog.php');
|
||||
require_once($this->INSTALL_PATH . '/../classes/general/Exception/InvalidJsonException.php');
|
||||
require_once($this->INSTALL_PATH . '/../classes/general/Exception/CurlException.php');
|
||||
require_once($this->INSTALL_PATH . '/../classes/general/RestNormalizer.php');
|
||||
require_once($this->INSTALL_PATH . '/../classes/general/Logger.php');
|
||||
require_once($this->INSTALL_PATH . '/../classes/general/services/RetailCrmService.php');
|
||||
require_once($this->INSTALL_PATH . '/../lib/component/constants.php');
|
||||
require_once($this->INSTALL_PATH . '/../classes/general/ApiClient_v5.php');
|
||||
require_once($this->INSTALL_PATH . '/../classes/general/order/RetailCrmOrder_v5.php');
|
||||
require_once($this->INSTALL_PATH . '/../classes/general/history/RetailCrmHistory_v5.php');
|
||||
require_once($this->INSTALL_PATH . '/../classes/general/cart/RetailCrmCart_v5.php');
|
||||
require_once($this->INSTALL_PATH . '/../lib/service/managerservice.php');
|
||||
require_once($this->INSTALL_PATH . '/../lib/service/loyaltyservice.php');
|
||||
require_once($this->INSTALL_PATH . '/../lib/service/loyaltyaccountservice.php');
|
||||
require_once($this->INSTALL_PATH . '/../lib/repository/managerrepository.php');
|
||||
require_once($this->INSTALL_PATH . '/../classes/general/services/BitrixOrderService.php');
|
||||
require_once($this->INSTALL_PATH . '/../lib/model/bitrix/abstractmodelproxy.php');
|
||||
require_once($this->INSTALL_PATH . '/../lib/model/bitrix/orderprops.php');
|
||||
require_once($this->INSTALL_PATH . '/../lib/model/bitrix/tomodule.php');
|
||||
require_once($this->INSTALL_PATH . '/../lib/repository/abstractrepository.php');
|
||||
require_once($this->INSTALL_PATH . '/../lib/repository/orderpropsrepository.php');
|
||||
require_once($this->INSTALL_PATH . '/../lib/repository/persontyperepository.php');
|
||||
require_once($this->INSTALL_PATH . '/../lib/repository/tomodulerepository.php');
|
||||
require_once($this->INSTALL_PATH . '/../lib/model/bitrix/orm/tomodule.php');
|
||||
require_once($this->INSTALL_PATH . '/../lib/model/bitrix/agreement.php');
|
||||
require_once($this->INSTALL_PATH . '/../lib/repository/agreementrepository.php');
|
||||
require_once($this->INSTALL_PATH . '/../lib/service/orderloyaltydataservice.php');
|
||||
require_once($this->INSTALL_PATH . '/../lib/service/currencyservice.php');
|
||||
require_once($this->INSTALL_PATH . '/../lib/component/factory/clientfactory.php');
|
||||
require_once($this->INSTALL_PATH . '/../lib/component/apiclient/clientadapter.php');
|
||||
require_once($this->INSTALL_PATH . '/../lib/component/advanced/loyaltyinstaller.php');
|
|
@ -1,6 +1,6 @@
|
|||
<?php
|
||||
|
||||
$arModuleVersion = [
|
||||
'VERSION' => '6.5.32',
|
||||
'VERSION_DATE' => '2024-10-03 17:00:00'
|
||||
'VERSION' => '6.6.11',
|
||||
'VERSION_DATE' => '2025-03-26 16:00:00'
|
||||
];
|
||||
|
|
|
@ -47,3 +47,5 @@ $MESS['BASE_PRICE'] = 'Base price';
|
|||
$MESS['WAIT'] = 'Loading...';
|
||||
$MESS["OFFERS_VALUE"] = "Maximum number of trade offers for a product";
|
||||
$MESS["LOAD_NON_ACTIVITY"] = "Unload inactive products, services and trade offers";
|
||||
$MESS["ADD_PROPERTY"] = "Add property";
|
||||
$MESS["DELETE_PROPERTY"] = "Delete";
|
||||
|
|
|
@ -23,6 +23,7 @@ $MESS ['ORDER_CUSTOM'] = 'Custom fields';
|
|||
$MESS ['COUPON_CUSTOM_FIELD'] = 'Select a custom field in the CRM to transfer the applied coupon in the Bitrix order';
|
||||
$MESS ['SELECT_VALUE'] = '-- Select a value --';
|
||||
$MESS ['ORDER_UPLOAD'] = 'Re-upload orders';
|
||||
$MESS ['LP_WARNING'] = 'Loyalty program of RetailCRM is available only with the “Uploading orders by event” options active';
|
||||
$MESS ['ORDER_NUMBER'] = 'Order numbers: ';
|
||||
$MESS ['ORDER_UPLOAD_INFO'] = 'Click "Start uploading" to upload all the orders . Or list the required order IDs separated by commas, intervals or dashes. For example: 1, 3, 5-10, 12, 13... etc.';
|
||||
$MESS ['ICRM_OPTIONS_SUBMIT_TITLE'] = 'Save settings';
|
||||
|
@ -41,6 +42,7 @@ $MESS ['ICRM_OPTIONS_ORDER_DISCHARGE_TAB'] = 'Orders uploading mode';
|
|||
$MESS ['ORDER_DISCH'] = 'Orders uploading mode';
|
||||
$MESS ['DISCHARGE_AGENT'] = 'Upload orders using agent';
|
||||
$MESS ['DISCHARGE_EVENTS'] = 'Upload orders by events';
|
||||
$MESS ['DISCHARGE_WITHOUT_UPDATE'] = 'Upload orders by agent (without update)';
|
||||
$MESS ['INFO_2'] = ' Set the correspondence between 1C-Bitrix and RetailCRM order fields.';
|
||||
$MESS ['ORDER_PROPS'] = 'Correspondence settings between RetailCRM order fields and 1C-Bitrix order properties';
|
||||
$MESS ['FIO'] = 'Full name';
|
||||
|
@ -146,7 +148,7 @@ $MESS ['NO_INTEGRATION_PAYMENT'] = '(Non-integrated)';
|
|||
$MESS ['ERR_CHECK_JOURNAL'] = 'Error while saving. Details in the event log';
|
||||
$MESS ['ERROR_LINK_INTEGRATION_PAYMENT'] = 'Error in comparing integration payments';
|
||||
$MESS ['ERROR_UPDATE_PAYMENT_TYPES_DELIVERY'] = 'Error when updating payment methods for deliveries';
|
||||
$MESS ['INTEGRATION_PAYMENT_LABEL'] = 'When correlating CMS and CRM integration payments, a regular payment is created on the system side to which orders will be linked.';
|
||||
$MESS ['INTEGRATION_PAYMENT_LABEL'] = 'When correlating CMS and CRM integration payments, a regular payment is created on the system side to which orders will be linked. <br> If you have integration deliveries in your CRM, then you need to enable the new payment method in the integration settings';
|
||||
$MESS ['NEED_PERMISSIONS_REFERENCE_LABEL'] = 'For this option to work correctly, the api key needs access to receive and edit reference book';
|
||||
|
||||
$MESS ['FIX_UPLOAD_CUSTOMER_HEADER'] = 'Fix customer registration date in CRM';
|
||||
|
|
|
@ -47,3 +47,5 @@ $MESS["UNIT_MEASUREMENT_KG"] = "кг.";
|
|||
$MESS['BASE_PRICE'] = 'Базовая цена';
|
||||
$MESS['WAIT'] = 'Загрузка...';
|
||||
$MESS["OFFERS_VALUE"] = "Максимальное количество торговых предложений у товара";
|
||||
$MESS["ADD_PROPERTY"] = "Добавить свойство";
|
||||
$MESS["DELETE_PROPERTY"] = "Удалить";
|
||||
|
|
|
@ -21,6 +21,7 @@ $MESS ['PAYMENT_TYPES_LIST'] = 'Способы оплаты';
|
|||
$MESS ['PAYMENT_STATUS_LIST'] = 'Статусы';
|
||||
$MESS ['ORDER_TYPES_LIST'] = 'Типы заказа';
|
||||
$MESS ['CRM_ORDER_METHODS'] = 'Передача заказов из CRM в Битрикс';
|
||||
$MESS ['LP_WARNING'] = 'Программа лояльности RetailCRM доступна только при активной опции "Выгрузка заказов по событию"';
|
||||
$MESS ['CRM_ORDER_METHODS_OPTION'] = 'Выгружать из RetailCRM заказы оформленные выбранными способами';
|
||||
$MESS ['CONTRAGENTS_TYPES_LIST'] = 'Тип контрагента';
|
||||
$MESS ['PAYMENT_LIST'] = 'Оплата';
|
||||
|
@ -62,7 +63,7 @@ $MESS ['ICRM_OPTIONS_ORDER_DISCHARGE_TAB'] = 'Режим выгрузки зак
|
|||
$MESS ['ORDER_DISCH'] = 'Режим выгрузки заказов';
|
||||
$MESS ['DISCHARGE_AGENT'] = 'Выгрузка заказов с помощью агента';
|
||||
$MESS ['DISCHARGE_EVENTS'] = 'Выгрузка заказов по событию';
|
||||
|
||||
$MESS ['DISCHARGE_WITHOUT_UPDATE'] = 'Выгрузка заказов по агенту (только создание заказов)';
|
||||
$MESS ['INFO_2'] = ' Задайте соответствие между полями заказа 1C-Битрикс и RetailCRM.';
|
||||
|
||||
$MESS ['ORDER_PROPS'] = 'Настройки соответствия полей заказа RetailCRM свойствам заказа 1С-Битрикс';
|
||||
|
@ -212,7 +213,7 @@ $MESS ['NO_INTEGRATION_PAYMENT'] = '(Не интеграционная)';
|
|||
$MESS ['ERR_CHECK_JOURNAL'] = 'Ошибка при сохранении. Подробности в журнале событий';
|
||||
$MESS ['ERROR_LINK_INTEGRATION_PAYMENT'] = 'Ошибка при сопоставлении интеграционных оплат';
|
||||
$MESS ['ERROR_UPDATE_PAYMENT_TYPES_DELIVERY'] = 'Ошибка при обновлении способов оплаты для доставок';
|
||||
$MESS ['INTEGRATION_PAYMENT_LABEL'] = 'При сопоставлении интеграционных оплат CRM, на стороне системы создаётся обычная оплата, к которой будут привязываться заказы.';
|
||||
$MESS ['INTEGRATION_PAYMENT_LABEL'] = 'При сопоставлении интеграционных оплат CRM, на стороне системы создаётся обычная оплата, к которой будут привязываться заказы. <br> Если в вашей CRM используются интеграционные доставки, новый способ оплаты необходимо вручную активировать в настройках интеграций.';
|
||||
$MESS ['NEED_PERMISSIONS_REFERENCE_LABEL'] = 'Для корректной работы опции апи-ключу необходимы доступы на получение и редактирование справочников';
|
||||
|
||||
$MESS ['FIX_UPLOAD_CUSTOMER_HEADER'] = 'Исправление даты регистрации клиентов в CRM';
|
||||
|
|
|
@ -24,15 +24,19 @@ trait InstallerTrait
|
|||
false
|
||||
);
|
||||
|
||||
$path = $_SERVER['DOCUMENT_ROOT'] . '/local/';
|
||||
$pathFrom = $_SERVER['DOCUMENT_ROOT'] .
|
||||
'/bitrix/modules/' .
|
||||
Constants::MODULE_ID .
|
||||
'/install/export/bitrix/js/intaro/export'
|
||||
;
|
||||
|
||||
CheckDirPath($path);
|
||||
|
||||
$file = new \Bitrix\Main\IO\File($path . 'icml_property_retailcrm.txt', $siteId = null);
|
||||
|
||||
if (!$file->isExists()) {
|
||||
$file->putContents("");
|
||||
}
|
||||
CopyDirFiles(
|
||||
$pathFrom,
|
||||
$_SERVER['DOCUMENT_ROOT'] . '/bitrix/js/intaro/export/',
|
||||
true,
|
||||
true,
|
||||
false
|
||||
);
|
||||
}
|
||||
|
||||
public function subscriptionSetup()
|
||||
|
|
|
@ -161,6 +161,7 @@ class LoyaltyInstaller
|
|||
'EDIT_FORM_LABEL' => ['ru' => $filed['title']],
|
||||
|
||||
];
|
||||
|
||||
$props = array_merge($arProps, $customProps);
|
||||
$obUserField = new CUserTypeEntity();
|
||||
$dbRes = CUserTypeEntity::GetList([], ['FIELD_NAME' => $filed['name']])->fetch();
|
||||
|
@ -228,7 +229,7 @@ class LoyaltyInstaller
|
|||
$eventManager->unRegisterEventHandler(
|
||||
$event['FROM_MODULE'],
|
||||
$event['EVENT_NAME'],
|
||||
$this->MODULE_ID,
|
||||
Constants::MODULE_ID,
|
||||
EventsHandlers::class,
|
||||
$event['EVENT_NAME'].'Handler'
|
||||
);
|
||||
|
|
|
@ -548,6 +548,36 @@ class ConfigProvider
|
|||
return static::getOption(Constants::CRM_ORDER_LAST_ID);
|
||||
}
|
||||
|
||||
/**
|
||||
* getLastOrderUpdate
|
||||
*
|
||||
* @return bool|string|null
|
||||
*/
|
||||
public static function getLastOrderUpdate()
|
||||
{
|
||||
return static::getOption(Constants::LAST_ORDER_UPDATE);
|
||||
}
|
||||
|
||||
/**
|
||||
* setLastOrderUpdate
|
||||
*
|
||||
* @return bool|string|null
|
||||
*/
|
||||
public static function setLastOrderUpdate($value)
|
||||
{
|
||||
static::setOption(Constants::LAST_ORDER_UPDATE, $value);
|
||||
}
|
||||
|
||||
/**
|
||||
* getOrderDischarge
|
||||
*
|
||||
* @return bool|string|null
|
||||
*/
|
||||
public static function getOrderDischarge()
|
||||
{
|
||||
return static::getOption(Constants::CRM_ORDER_DISCHARGE);
|
||||
}
|
||||
|
||||
/**
|
||||
* getSendPaymentAmount
|
||||
*
|
||||
|
|
|
@ -18,6 +18,7 @@ namespace Intaro\RetailCrm\Component;
|
|||
*/
|
||||
class Constants
|
||||
{
|
||||
public const MODULE_VERSION = '6.6.11';
|
||||
public const CRM_PURCHASE_PRICE_NULL = 'purchasePrice_null';
|
||||
public const BITRIX_USER_ID_PREFIX = 'bitrixUserId-';
|
||||
public const CRM_USERS_MAP = 'crm_users_map';
|
||||
|
@ -133,4 +134,5 @@ class Constants
|
|||
public const MODULE_DEACTIVATE = 'module_deactivate';
|
||||
public const AGENTS_DEACTIVATE = 'agents_deactivate';
|
||||
public const EVENTS_DEACTIVATE = 'events_deactivate';
|
||||
public const LAST_ORDER_UPDATE = 'last_order_update';
|
||||
}
|
||||
|
|
|
@ -83,7 +83,8 @@ class EventsHandlers
|
|||
|
||||
$isWriteOffAvailable = $bonusInput > 0
|
||||
&& $availableBonuses > 0
|
||||
&& $jsDataTotal['ORDER_TOTAL_PRICE'] >= $bonusDiscount + $loyaltyDiscountInput;
|
||||
&& $jsDataTotal['ORDER_TOTAL_PRICE'] >= $bonusDiscount + $loyaltyDiscountInput
|
||||
;
|
||||
|
||||
if ($isWriteOffAvailable || $loyaltyDiscountInput > 0) {
|
||||
$jsDataTotal['ORDER_TOTAL_PRICE']
|
||||
|
@ -132,9 +133,6 @@ class EventsHandlers
|
|||
try {
|
||||
/** @var Order $order */
|
||||
$order = $event->getParameter('ENTITY');
|
||||
|
||||
$saveResult = RetailCrmEvent::orderSave($order);
|
||||
Utils::handleApiErrors($saveResult);
|
||||
|
||||
$isBonusInput = (
|
||||
!empty($_POST['bonus-input'])
|
||||
|
|
95
intaro.retailcrm/lib/controller/customexportprops.php
Normal file
95
intaro.retailcrm/lib/controller/customexportprops.php
Normal file
|
@ -0,0 +1,95 @@
|
|||
<?php
|
||||
|
||||
namespace Intaro\RetailCrm\Controller;
|
||||
|
||||
use Bitrix\Main\Engine\Controller;
|
||||
use Bitrix\Main\Result;
|
||||
use Bitrix\Main\Error;
|
||||
use Bitrix\Main\Application;
|
||||
use Intaro\RetailCrm\Icml\SettingsService;
|
||||
|
||||
/**
|
||||
* @category Integration
|
||||
* @package Intaro\RetailCrm\Controller
|
||||
* @author RetailCRM <integration@retailcrm.ru>
|
||||
* @license MIT
|
||||
* @link http://retailcrm.ru
|
||||
* @see http://retailcrm.ru/docs
|
||||
*/
|
||||
class CustomExportProps extends Controller
|
||||
{
|
||||
private function getRequestData(): array
|
||||
{
|
||||
$data = $this->getRequest()->getInput();
|
||||
|
||||
if ($data === null) {
|
||||
|
||||
}
|
||||
|
||||
return json_decode($data, true);
|
||||
}
|
||||
|
||||
public function saveAction()
|
||||
{
|
||||
$requestData = $this->getRequestData();
|
||||
$props = $requestData['properties'];
|
||||
$profileId = $requestData['profileId'];
|
||||
|
||||
$settingsService = SettingsService::getInstance(
|
||||
[],
|
||||
null,
|
||||
$profileId
|
||||
);
|
||||
|
||||
$idCategories = array_keys($props);
|
||||
|
||||
if ($idCategories !== []) {
|
||||
$settingsService->setProfileCatalogs($idCategories);//сразу заменяет и create и update
|
||||
}
|
||||
|
||||
foreach ($props as $catalogId => $propsArray) {
|
||||
$catalogCustomProps = [];
|
||||
|
||||
foreach ($propsArray as $property) {
|
||||
$catalogCustomProps[] = [
|
||||
'code' => $property['code'],
|
||||
'title' => $property['title']
|
||||
];
|
||||
}
|
||||
|
||||
$settingsService
|
||||
->setCatalogCustomPropsOptionName($catalogId)
|
||||
->saveCustomProps($catalogCustomProps)
|
||||
;
|
||||
}
|
||||
}
|
||||
|
||||
public function deleteAction()
|
||||
{
|
||||
$requestData = $this->getRequestData();
|
||||
$props = $requestData['properties'];
|
||||
$profileId = $requestData['profileId'];
|
||||
|
||||
$settingsService = SettingsService::getInstance(
|
||||
[],
|
||||
null,
|
||||
$profileId
|
||||
);
|
||||
|
||||
foreach ($props as $catalogId => $propsArray) {
|
||||
$catalogCustomProps = [];
|
||||
|
||||
foreach ($propsArray as $property) {
|
||||
$catalogCustomProps[] = [
|
||||
'code' => $property['code'],
|
||||
'title' => $property['title']
|
||||
];
|
||||
}
|
||||
|
||||
$settingsService
|
||||
->setCatalogCustomPropsOptionName($catalogId)
|
||||
->removeCustomProps($catalogCustomProps, $catalogId)
|
||||
;
|
||||
}
|
||||
}
|
||||
}
|
|
@ -163,7 +163,7 @@ class IcmlWriter
|
|||
$this->writeOptionalSimpleElement('vatRate', $offer->vatRate);
|
||||
$this->writeOptionalSimpleElement('weight', $offer->weight);
|
||||
$this->writeOptionalSimpleElement('dimensions', $offer->dimensions);
|
||||
$this->writeOptionalSimpleElement('purchasePrice', $offer->purchasePrice);
|
||||
$this->writeOptionalPurchasePrice($offer->purchasePrice);
|
||||
$this->writer->endElement();
|
||||
}
|
||||
|
||||
|
@ -180,6 +180,16 @@ class IcmlWriter
|
|||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Запись закупочной стоимости
|
||||
*
|
||||
* @param $value
|
||||
*/
|
||||
private function writeOptionalPurchasePrice($value): void
|
||||
{
|
||||
$this->writeSimpleElement('purchasePrice', !empty($value) ? $value : 0);
|
||||
}
|
||||
|
||||
/**
|
||||
* @param string $name
|
||||
* @param $value
|
||||
|
|
|
@ -8,14 +8,19 @@ use CCatalogGroup;
|
|||
use CCatalogSku;
|
||||
use CCatalogVat;
|
||||
use CIBlock;
|
||||
use COption;
|
||||
use Bitrix\Main\Config\Option;
|
||||
use Intaro\RetailCrm\Service\Hl;
|
||||
use RetailcrmConfigProvider;
|
||||
use Bitrix\Main\Application;
|
||||
use Bitrix\Main\Entity\Query;
|
||||
|
||||
/**
|
||||
* Отвечает за управление настройками выгрузки icml каталога
|
||||
*
|
||||
* @var $PROFILE_ID - зачем если получаем в в конструкторе инстансе
|
||||
* Class SettingsService
|
||||
*
|
||||
*
|
||||
* @package Intaro\RetailCrm\Icml
|
||||
*/
|
||||
class SettingsService
|
||||
|
@ -32,6 +37,14 @@ class SettingsService
|
|||
*/
|
||||
public const INFOBLOCK_WITH_SKU = 'P';
|
||||
|
||||
private const MODULE_ID = 'intaro.retailcrm';
|
||||
|
||||
private string $catalogCustomPropsOptionName;
|
||||
|
||||
private string $profileCatalogsOptionName;
|
||||
|
||||
private string $exportProfileId;
|
||||
|
||||
/**
|
||||
* @var array
|
||||
*/
|
||||
|
@ -98,7 +111,13 @@ class SettingsService
|
|||
public $loadNonActivity;
|
||||
|
||||
/** @var array */
|
||||
public $actrualPropList = [];
|
||||
public $actualPropList = [];
|
||||
|
||||
/** @var array */
|
||||
public $customPropList = [];
|
||||
|
||||
/** @var array */
|
||||
public $defaultPropList = [];
|
||||
|
||||
/**
|
||||
* @var \Intaro\RetailCrm\Icml\SettingsService|null
|
||||
|
@ -112,7 +131,7 @@ class SettingsService
|
|||
* @param array $arOldSetupVars
|
||||
* @param string|null $action
|
||||
*/
|
||||
private function __construct(array $arOldSetupVars, ?string $action)
|
||||
private function __construct(array $arOldSetupVars, ?string $action, ?string $profileId)
|
||||
{
|
||||
$this->arOldSetupVars = $arOldSetupVars;
|
||||
$this->action = $action;
|
||||
|
@ -128,7 +147,15 @@ class SettingsService
|
|||
$this->getPriceTypes();
|
||||
$this->getVatRates();
|
||||
|
||||
$this->actrualPropList = array_merge($this->getIblockPropsPreset(), $this->parseNewProps());
|
||||
$this->exportProfileId = $profileId ?? '0';
|
||||
$this->profileCatalogsOptionName = sprintf('exportProfileId_%s_catalogs', $this->exportProfileId);
|
||||
|
||||
$this->linkNewProfile();
|
||||
$this->deleteEmptyProfileCatalogs();
|
||||
|
||||
$this->customPropList = $this->getNewProps();
|
||||
$this->defaultPropList = $this->getIblockPropsPreset();
|
||||
$this->actualPropList = $this->getActualPropList();
|
||||
}
|
||||
|
||||
/**
|
||||
|
@ -137,16 +164,29 @@ class SettingsService
|
|||
*
|
||||
* @return \Intaro\RetailCrm\Icml\SettingsService|null
|
||||
*/
|
||||
public static function getInstance(array $arOldSetupVars, ?string $action): ?SettingsService
|
||||
public static function getInstance(array $arOldSetupVars, ?string $action, ?string $profileId): ?SettingsService
|
||||
{
|
||||
if (is_null(self::$instance)) {
|
||||
self::$instance = new self($arOldSetupVars, $action);
|
||||
self::$instance = new self($arOldSetupVars, $action, $profileId);
|
||||
}
|
||||
|
||||
return self::$instance;
|
||||
}
|
||||
|
||||
public function getPriceTypes()
|
||||
private function getActualPropList(): array
|
||||
{
|
||||
$customProps = [];
|
||||
|
||||
foreach ($this->customPropList as $propsByCatalog) {
|
||||
foreach ($propsByCatalog as $code => $value) {
|
||||
$customProps[$code] = $value;
|
||||
}
|
||||
}
|
||||
|
||||
return array_merge($this->defaultPropList, $customProps);
|
||||
}
|
||||
|
||||
private function getPriceTypes()
|
||||
{
|
||||
$dbPriceType = CCatalogGroup::GetList(['SORT' => 'ASC'], [], [], [], ['ID', 'NAME', 'BASE']);
|
||||
|
||||
|
@ -155,7 +195,7 @@ class SettingsService
|
|||
}
|
||||
}
|
||||
|
||||
public function getVatRates()
|
||||
private function getVatRates()
|
||||
{
|
||||
$dbVatRate = CCatalogVat::GetListEx(['SORT' => 'ASC'], ['ACTIVE' => 'Y'], false, false, ['ID', 'NAME', 'RATE']);
|
||||
|
||||
|
@ -230,7 +270,9 @@ class SettingsService
|
|||
private function setProperties(array &$properties, string $propName): void
|
||||
{
|
||||
foreach ($this->arOldSetupVars[$propName] as $iblock => $val) {
|
||||
$properties[$iblock][$propName] = $val;
|
||||
if (!empty($val)) {
|
||||
$properties[$iblock][$propName] = $val;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -260,27 +302,22 @@ class SettingsService
|
|||
];
|
||||
}
|
||||
|
||||
private function parseNewProps(): array
|
||||
private function getNewProps(): array
|
||||
{
|
||||
global $APPLICATION;
|
||||
|
||||
$result = [];
|
||||
$text = $APPLICATION->GetFileContent($_SERVER["DOCUMENT_ROOT"] . "/local/icml_property_retailcrm.txt");
|
||||
$currentProfileCatalogIds = $this->getProfileCatalogs();
|
||||
|
||||
if ($text === false) {
|
||||
return $result;
|
||||
}
|
||||
if (!is_null($currentProfileCatalogIds)) {
|
||||
foreach ($currentProfileCatalogIds as $catalogId) {
|
||||
$catalogCustomProps = $this
|
||||
->setCatalogCustomPropsOptionName($catalogId)
|
||||
->getCustomProps()
|
||||
;
|
||||
|
||||
preg_match_all('/\w+\s*=\s*\w+[ *\w+]*/mu', $text, $matches);
|
||||
|
||||
foreach ($matches[0] as $newProp) {
|
||||
$elements = explode("=", $newProp);
|
||||
|
||||
if (empty($elements[0]) || empty($elements[1])) {
|
||||
continue;
|
||||
foreach ($catalogCustomProps as $prop) {
|
||||
$result[$catalogId][$prop['code']] = $prop['title'];
|
||||
}
|
||||
}
|
||||
|
||||
$result[trim($elements[0])] = trim($elements[1]);
|
||||
}
|
||||
|
||||
return $result;
|
||||
|
@ -289,7 +326,7 @@ class SettingsService
|
|||
/**
|
||||
* @return array
|
||||
*/
|
||||
public function getHintProps(): array
|
||||
private function getHintProps(): array
|
||||
{
|
||||
return [
|
||||
'article' => ['ARTICLE', 'ART', 'ARTNUMBER', 'ARTICUL', 'ARTIKUL'],
|
||||
|
@ -365,7 +402,7 @@ class SettingsService
|
|||
|
||||
public function setProps(): void
|
||||
{
|
||||
foreach (array_keys($this->actrualPropList) as $prop) {
|
||||
foreach (array_keys($this->actualPropList) as $prop) {
|
||||
$this->setProperties($this->iblockPropertySku, 'iblockPropertySku_' . $prop);
|
||||
$this->setProperties($this->iblockPropertyUnitSku, 'iblockPropertyUnitSku_' . $prop);
|
||||
$this->setProperties($this->iblockPropertyProduct, 'iblockPropertyProduct_' . $prop);
|
||||
|
@ -548,7 +585,7 @@ class SettingsService
|
|||
*
|
||||
* @return array
|
||||
*/
|
||||
public function getSiteList(int $iblockId): array
|
||||
private function getSiteList(int $iblockId): array
|
||||
{
|
||||
$siteList = [];
|
||||
|
||||
|
@ -576,7 +613,9 @@ class SettingsService
|
|||
$dbSkuProperties = CIBlock::GetProperties($iblockOffer['IBLOCK_ID'], [], ['MULTIPLE' => 'N']);
|
||||
|
||||
while ($prop = $dbSkuProperties->Fetch()) {
|
||||
$propertiesSKU[] = $prop;
|
||||
if ($prop['CODE'] !== '') {
|
||||
$propertiesSKU[] = $prop;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -597,7 +636,7 @@ class SettingsService
|
|||
$props = [];
|
||||
|
||||
if (isset($oldValues[$iblockId])) {
|
||||
foreach (array_keys($this->actrualPropList) as $prop) {
|
||||
foreach (array_keys($this->actualPropList) as $prop) {
|
||||
$fullKey = $keyGroup . '_' . $prop;
|
||||
$props[$prop] = $oldValues[$iblockId][$fullKey];
|
||||
}
|
||||
|
@ -623,14 +662,16 @@ class SettingsService
|
|||
*
|
||||
* @return array|null
|
||||
*/
|
||||
public function getProductProps(int $iblockId): ?array
|
||||
private function getProductProps(int $iblockId): ?array
|
||||
{
|
||||
$propertiesProduct = null;
|
||||
|
||||
$iblockResult = CIBlock::GetProperties($iblockId, [], ['MULTIPLE' => 'N']);
|
||||
|
||||
while ($prop = $iblockResult->Fetch()) {
|
||||
$propertiesProduct[] = $prop;
|
||||
if ($prop['CODE'] !== '') {
|
||||
$propertiesProduct[] = $prop;
|
||||
}
|
||||
}
|
||||
|
||||
return $propertiesProduct;
|
||||
|
@ -642,7 +683,7 @@ class SettingsService
|
|||
*
|
||||
* @return bool
|
||||
*/
|
||||
public function isExport($iblockId, $iblockExport): bool
|
||||
private function isExport($iblockId, $iblockExport): bool
|
||||
{
|
||||
if (is_array($iblockExport) && count($iblockExport) !== 0) {
|
||||
return (in_array($iblockId, $iblockExport));
|
||||
|
@ -747,4 +788,152 @@ class SettingsService
|
|||
|
||||
return [$arIBlockList, $intCountChecked, $intCountAvailIBlock, $arIBlockList['iblockExport'] ?? false];
|
||||
}
|
||||
|
||||
public function setCatalogCustomPropsOptionName(string $catalogId): self
|
||||
{
|
||||
$this->catalogCustomPropsOptionName = sprintf(
|
||||
'exportCustomProps_ProfileId_%s_catalogId_%s',
|
||||
$this->exportProfileId,
|
||||
$catalogId
|
||||
);
|
||||
|
||||
return $this;
|
||||
}
|
||||
|
||||
private function getCustomProps(): ?array
|
||||
{
|
||||
$props = unserialize(COption::GetOptionString(self::MODULE_ID, $this->catalogCustomPropsOptionName));
|
||||
|
||||
if (!$props) {
|
||||
return null;
|
||||
}
|
||||
|
||||
return $props;
|
||||
}
|
||||
|
||||
public function removeCustomProps(array $propsToDelete, string $catalogId): void
|
||||
{
|
||||
$currentCatalogProps = $this->getCustomProps();
|
||||
$updatedCatalogProps = array_values(
|
||||
array_filter(
|
||||
$currentCatalogProps,
|
||||
fn ($currentProp) => !in_array($currentProp, $propsToDelete)
|
||||
));
|
||||
|
||||
if (empty($updatedCatalogProps)) {
|
||||
$this->deleteOptionEntry($this->catalogCustomPropsOptionName);
|
||||
$this->deleteProfileCatalog($catalogId);
|
||||
} else {
|
||||
$this->updateCustomProps($updatedCatalogProps);
|
||||
}
|
||||
}
|
||||
|
||||
private function updateCustomProps(array $updatedProps)
|
||||
{
|
||||
$serializedProps = serialize($updatedProps);
|
||||
$this->updateOptionEntry($this->catalogCustomPropsOptionName, $serializedProps);
|
||||
}
|
||||
|
||||
private function setCustomProps(array $props)
|
||||
{
|
||||
$propsString = serialize($props);
|
||||
$this->setOptionEntry($this->catalogCustomPropsOptionName, $propsString);
|
||||
}
|
||||
|
||||
public function saveCustomProps(array $newProps): void
|
||||
{
|
||||
$currentProps = $this->getCustomProps();
|
||||
|
||||
if (is_null($currentProps)) {
|
||||
$this->setCustomProps($newProps);
|
||||
} else {
|
||||
$updatedProps = array_merge($currentProps, $newProps);
|
||||
$this->updateCustomProps($updatedProps);
|
||||
}
|
||||
}
|
||||
|
||||
private function getProfileCatalogs(): ?array
|
||||
{
|
||||
$catalogs = unserialize(COption::GetOptionString(self::MODULE_ID, $this->profileCatalogsOptionName));
|
||||
|
||||
if (!$catalogs) {
|
||||
return null;
|
||||
}
|
||||
|
||||
return $catalogs;
|
||||
}
|
||||
|
||||
public function setProfileCatalogs(array $catalogsId): void
|
||||
{
|
||||
$catalogs = serialize($catalogsId);
|
||||
$this->setOptionEntry($this->profileCatalogsOptionName, $catalogs);
|
||||
}
|
||||
|
||||
private function deleteProfileCatalog(string $catalogId): void
|
||||
{
|
||||
$currentCatalogs = $this->getProfileCatalogs();
|
||||
$catalogIdIndex = array_search($catalogId, $currentCatalogs);
|
||||
|
||||
if ($catalogIdIndex !== false) {
|
||||
unset($currentCatalogs[$catalogIdIndex]);
|
||||
}
|
||||
|
||||
$updatedCatalogs = serialize($currentCatalogs);
|
||||
$this->updateOptionEntry($this->profileCatalogsOptionName, $updatedCatalogs);
|
||||
}
|
||||
|
||||
private function deleteEmptyProfileCatalogs(): void
|
||||
{
|
||||
$currentCatalogs = $this->getProfileCatalogs();
|
||||
|
||||
if (is_null($currentCatalogs)) {
|
||||
$this->deleteOptionEntry($this->profileCatalogsOptionName);
|
||||
}
|
||||
}
|
||||
|
||||
private function setOptionEntry(string $name, string $value)
|
||||
{
|
||||
COption::SetOptionString(self::MODULE_ID, $name, $value);
|
||||
}
|
||||
|
||||
private function updateOptionEntry(string $name, string $value)
|
||||
{
|
||||
COption::SetOptionString(self::MODULE_ID, $name, '');
|
||||
COption::SetOptionString(self::MODULE_ID, $name, $value);
|
||||
}
|
||||
|
||||
private function deleteOptionEntry(string $name)
|
||||
{
|
||||
COption::RemoveOption(self::MODULE_ID, $name);
|
||||
}
|
||||
|
||||
private function linkNewProfile(): void
|
||||
{
|
||||
$currentProfileCatalogs = unserialize(COption::GetOptionString(self::MODULE_ID, $this->profileCatalogsOptionName));
|
||||
|
||||
if (!$currentProfileCatalogs) {
|
||||
$tmpProfileName = 'exportProfileId_0_catalogs';
|
||||
$currentProfileCatalogs = unserialize(COption::GetOptionString(self::MODULE_ID, $tmpProfileName));
|
||||
|
||||
if ($currentProfileCatalogs) {
|
||||
$this->setOptionEntry($this->profileCatalogsOptionName, serialize($currentProfileCatalogs));
|
||||
$this->deleteOptionEntry($tmpProfileName);
|
||||
}
|
||||
}
|
||||
|
||||
foreach ($currentProfileCatalogs as $catalogId) {
|
||||
$optionName = sprintf('exportCustomProps_ProfileId_%s_catalogId_%s', $this->exportProfileId, $catalogId);
|
||||
$propsCatalog = unserialize(COption::GetOptionString(self::MODULE_ID, $optionName));
|
||||
|
||||
if (!$propsCatalog) {
|
||||
$tmpOptionName = sprintf('exportCustomProps_ProfileId_%s_catalogId_%s', '0', $catalogId);
|
||||
$propsCatalog = unserialize(COption::GetOptionString(self::MODULE_ID, $tmpOptionName));
|
||||
|
||||
if ($propsCatalog) {
|
||||
$this->setOptionEntry($optionName, serialize($propsCatalog));
|
||||
$this->deleteOptionEntry($tmpOptionName);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
|
@ -115,8 +115,8 @@ class XmlOfferBuilder
|
|||
public function __construct(XmlSetup $setup, array $measure, ?string $serverName)
|
||||
{
|
||||
$this->setup = $setup;
|
||||
$this->setSettingsService();
|
||||
$this->purchasePriceNull = RetailcrmConfigProvider::getCrmPurchasePrice();
|
||||
$this->settingsService = SettingsService::getInstance([], '');
|
||||
$this->vatRates = $this->settingsService->vatRates;
|
||||
$this->measures = $this->prepareMeasures($measure);
|
||||
$this->serverName = $serverName;
|
||||
|
@ -141,6 +141,12 @@ class XmlOfferBuilder
|
|||
return $this->xmlOffer;
|
||||
}
|
||||
|
||||
private function setSettingsService(): void
|
||||
{
|
||||
global $PROFILE_ID;
|
||||
$this->settingsService = SettingsService::getInstance([], '', $PROFILE_ID);
|
||||
}
|
||||
|
||||
/**
|
||||
* @param array $categories
|
||||
*/
|
||||
|
@ -479,7 +485,7 @@ class XmlOfferBuilder
|
|||
private function createParamObject(array $params): array
|
||||
{
|
||||
$offerParams = [];
|
||||
$names = $this->settingsService->actrualPropList;
|
||||
$names = $this->settingsService->actualPropList;
|
||||
|
||||
foreach ($params as $code => $value) {
|
||||
|
||||
|
|
|
@ -7,6 +7,7 @@ use Bitrix\Main\LoaderException;
|
|||
use Bitrix\Main\SystemException;
|
||||
use Bitrix\Main\UI\Extension;
|
||||
use Bitrix\Sale\Delivery\Services\Manager;
|
||||
use Bitrix\Sale\Internals\OrderTable;
|
||||
use Intaro\RetailCrm\Component\ApiClient\ClientAdapter;
|
||||
use Intaro\RetailCrm\Component\ConfigProvider;
|
||||
use Intaro\RetailCrm\Component\Constants;
|
||||
|
@ -22,6 +23,7 @@ IncludeModuleLangFile(__FILE__);
|
|||
include (__DIR__ . '/lib/component/advanced/loyaltyinstaller.php');
|
||||
|
||||
$mid = 'intaro.retailcrm';
|
||||
$loyaltyEventClass = 'Intaro\RetailCrm\Component\Handlers\EventsHandlers';
|
||||
$uri = $APPLICATION->GetCurPage() . '?mid=' . htmlspecialchars($mid) . '&lang=' . LANGUAGE_ID;
|
||||
|
||||
if (!CModule::IncludeModule('intaro.retailcrm') || !CModule::IncludeModule('sale') || !CModule::IncludeModule('iblock') || !CModule::IncludeModule('catalog')) {
|
||||
|
@ -85,7 +87,7 @@ if (method_exists(RCrmActions::class, 'customOrderPropList')
|
|||
}
|
||||
|
||||
//ajax update deliveryServices
|
||||
if (!empty($_SERVER['HTTP_X_REQUESTED_WITH']) && (strtolower($_SERVER['HTTP_X_REQUESTED_WITH']) === 'xmlhttprequest') && isset($_POST['ajax']) && ($_POST['ajax'] === 1)) {
|
||||
if (!empty($_SERVER['HTTP_X_REQUESTED_WITH']) && (strtolower($_SERVER['HTTP_X_REQUESTED_WITH']) === 'xmlhttprequest') && isset($_POST['ajax']) && ($_POST['ajax'] === '1')) {
|
||||
$api_host = COption::GetOptionString($mid, Constants::CRM_API_HOST_OPTION, 0);
|
||||
$api_key = COption::GetOptionString($mid, Constants::CRM_API_KEY_OPTION , 0);
|
||||
$api = new RetailCrm\ApiClient($api_host, $api_key);
|
||||
|
@ -108,7 +110,7 @@ if (!empty($_SERVER['HTTP_X_REQUESTED_WITH']) && (strtolower($_SERVER['HTTP_X_RE
|
|||
|
||||
foreach ($optionsDelivTypes as $key => $deliveryType) {
|
||||
foreach ($arDeliveryServiceAll as $deliveryService) {
|
||||
if ($deliveryService['PARENT_ID'] != 0 && $deliveryService['PARENT_ID'] === $key) {
|
||||
if ($deliveryService['PARENT_ID'] != 0 && $deliveryService['PARENT_ID'] == $key) {
|
||||
try {
|
||||
$api->deliveryServicesEdit(RCrmActions::clearArr([
|
||||
'code' => 'bitrix-' . $deliveryService['ID'],
|
||||
|
@ -314,16 +316,60 @@ if (isset($_POST['Update']) && ($_POST['Update'] === 'Y')) {
|
|||
//order discharge mode
|
||||
// 0 - agent
|
||||
// 1 - event
|
||||
// 2 - agent without update
|
||||
$orderDischarge = (int) htmlspecialchars(trim($_POST['order-discharge']));
|
||||
|
||||
if (($orderDischarge != $previousDischarge) && ($orderDischarge === 0)) {
|
||||
if (($orderDischarge != $previousDischarge) && ($orderDischarge === 0 || $orderDischarge === 2 )) {
|
||||
// remove depenedencies
|
||||
UnRegisterModuleDependences('sale', 'OnOrderUpdate', $mid, 'RetailCrmEvent', "onUpdateOrder");
|
||||
UnRegisterModuleDependences('sale', 'OnSaleOrderDeleted', $mid, 'RetailCrmEvent', "orderDelete");
|
||||
UnRegisterModuleDependences('sale', 'OnSaleOrderSaved', $mid, 'RetailCrmEvent', "orderSave");
|
||||
UnRegisterModuleDependences('sale', 'OnOrderUpdate', $mid, 'RetailCrmEvent', "onUpdateOrder");
|
||||
UnRegisterModuleDependences('sale', 'OnSaleOrderSaved', $mid, $loyaltyEventClass, 'OnSaleOrderSavedHandler');
|
||||
UnRegisterModuleDependences('sale', 'OnSaleComponentOrderResultPrepared', $mid, $loyaltyEventClass, 'OnSaleComponentOrderResultPreparedHandler');
|
||||
ConfigProvider::setLoyaltyProgramStatus('N');
|
||||
|
||||
$dateAgent = new DateTime();
|
||||
$intAgent = new DateInterval('PT60S');
|
||||
$dateAgent->add($intAgent);
|
||||
|
||||
CAgent::AddAgent(
|
||||
'RCrmActions::uploadOrdersAgent();',
|
||||
$mid,
|
||||
'N',
|
||||
180,
|
||||
$dateAgent->format('d.m.Y H:i:s'),
|
||||
'Y',
|
||||
$dateAgent->format('d.m.Y H:i:s'),
|
||||
30
|
||||
);
|
||||
|
||||
if ($orderDischarge === 0) {
|
||||
COption::SetOptionString($mid, Constants::LAST_ORDER_UPDATE, date("Y-m-d H:i:s"));
|
||||
} else {
|
||||
COption::RemoveOption($mid, Constants::LAST_ORDER_UPDATE);
|
||||
}
|
||||
|
||||
$dbOrder = OrderTable::GetList([
|
||||
'order' => ['ID' => 'DESC'],
|
||||
'limit' => 1,
|
||||
'select' => ['ID'],
|
||||
]);
|
||||
|
||||
$arOrder = $dbOrder->fetch();
|
||||
|
||||
if ($dbOrder) {
|
||||
COption::SetOptionString($mid, Constants::CRM_ORDER_LAST_ID, $arOrder['ID']);
|
||||
} else {
|
||||
COption::SetOptionString($mid, Constants::CRM_ORDER_LAST_ID, 0);
|
||||
}
|
||||
|
||||
} elseif (($orderDischarge != $previousDischarge) && ($orderDischarge === 1)) {
|
||||
// event dependencies
|
||||
RegisterModuleDependences('sale', 'OnOrderUpdate', $mid, 'RetailCrmEvent', "onUpdateOrder");
|
||||
RegisterModuleDependences('sale', 'OnSaleOrderDeleted', $mid, 'RetailCrmEvent', "orderDelete");
|
||||
RegisterModuleDependences('sale', 'OnSaleOrderSaved', $mid, 'RetailCrmEvent', "orderSave");
|
||||
COption::RemoveOption($mid, Constants::LAST_ORDER_UPDATE);
|
||||
CAgent::RemoveAgent("RCrmActions::uploadOrdersAgent();", $mid);
|
||||
}
|
||||
|
||||
$optionCart = COption::GetOptionString($mid, Constants::CART, 'N');
|
||||
|
@ -612,6 +658,8 @@ if (isset($_POST['Update']) && ($_POST['Update'] === 'Y')) {
|
|||
ConfigProvider::setLoyaltyProgramStatus('Y');
|
||||
} else {
|
||||
ConfigProvider::setLoyaltyProgramStatus('N');
|
||||
UnRegisterModuleDependences('sale', 'OnSaleOrderSaved', $mid, $loyaltyEventClass, 'OnSaleOrderSavedHandler');
|
||||
UnRegisterModuleDependences('sale', 'OnSaleComponentOrderResultPrepared', $mid, $loyaltyEventClass, 'OnSaleComponentOrderResultPreparedHandler');
|
||||
}
|
||||
|
||||
try {
|
||||
|
@ -829,7 +877,9 @@ if (isset($_POST['Update']) && ($_POST['Update'] === 'Y')) {
|
|||
$substitutionPaymentList[$integrationPayment] = $codePayment;
|
||||
|
||||
foreach ($originalPayment['deliveryTypes'] as $codeDelivery) {
|
||||
if (!isset($arResult['deliveryTypesList'][$codeDelivery])) {
|
||||
if (!isset($arResult['deliveryTypesList'][$codeDelivery]) ||
|
||||
isset($arResult['deliveryTypesList'][$codeDelivery]['integrationCode'])
|
||||
) {
|
||||
continue;
|
||||
}
|
||||
|
||||
|
@ -3007,9 +3057,17 @@ if (isset($_POST['Update']) && ($_POST['Update'] === 'Y')) {
|
|||
<label><input class="addr" type="radio" name="order-discharge" value="0" <?php if ($optionsDischarge === 0) {
|
||||
echo "checked";
|
||||
} ?>><?php echo GetMessage('DISCHARGE_AGENT'); ?></label>
|
||||
<label><input class="addr" type="radio" name="order-discharge" value="2" <?php if ($optionsDischarge === 2) {
|
||||
echo "checked";
|
||||
} ?>><?php echo GetMessage('DISCHARGE_WITHOUT_UPDATE'); ?></label>
|
||||
</b>
|
||||
</td>
|
||||
</tr>
|
||||
<tr>
|
||||
<td colspan="2" class="option-head option-other-top option-other-bottom" style="color:#c24141">
|
||||
<b><?php echo GetMessage('LP_WARNING'); ?></b>
|
||||
</td>
|
||||
</tr>
|
||||
<tr>
|
||||
<td colspan="2" class="option-head option-other-top option-other-bottom">
|
||||
<b><?php echo GetMessage('COUPON_CUSTOM_FIELD'); ?></b>
|
||||
|
|
1268
intaro.retailcrm/updater.php
Normal file → Executable file
1268
intaro.retailcrm/updater.php
Normal file → Executable file
File diff suppressed because it is too large
Load diff
|
@ -17,25 +17,6 @@ class SettingServiceTest extends \BitrixTestCase
|
|||
;
|
||||
}
|
||||
|
||||
public function testConstruct(): SettingsService
|
||||
{
|
||||
$path = $_SERVER['DOCUMENT_ROOT'] . '/local/';
|
||||
|
||||
CheckDirPath($path);
|
||||
|
||||
$file = new \Bitrix\Main\IO\File($path . '/icml_property_retailcrm.txt', $siteId = null);
|
||||
|
||||
$file->putContents("property1 = test prop \n property2 = test prop 2");
|
||||
|
||||
$settingService = SettingsService::getInstance($this->getSetupVars(), "");
|
||||
|
||||
$this->assertInstanceOf(SettingsService::class, $settingService);
|
||||
$this->assertArrayHasKey('property1', $settingService->actrualPropList);
|
||||
$this->assertArrayHasKey('property2', $settingService->actrualPropList);
|
||||
|
||||
return $settingService;
|
||||
}
|
||||
|
||||
private function getSetupVars()
|
||||
{
|
||||
return [
|
||||
|
|
Loading…
Add table
Reference in a new issue