то ждем ваше обращение в нашей службе тех поддержки.
Работа с пользовательскими свойствами инфоблоков
Примеры решения задач, возникающих при работе с элементами, разделами и свойствами инфоблоков.
Задача 1:
Получить значения всех свойств элемента, зная его ID.
1 <? $db_props = CIBlockElement::GetProperty(IBLOCK_ID, ELEMENT_ID, "sort", "asc", array()); 2 $PROPS = array(); 3 while($ar_props = $db_props->GetNext()) 4 $PROPS[$ar_props['CODE']] = $ar_props['VALUE'];?>
Теперь символьный код свойства является ключом ассоциативного массива $PROPS
, то есть, если вам нужно значение свойства с кодом price, то оно будет храниться в $PROPS['price']
.
Задача 2:
Получить свойства элементов, используя метод CIBlockElement::GetList
1 <? $arSelect = array("ID", "NAME", "PROPERTY_prop_code_1", "PROPERTY_prop_code_2"); 2 $res = CIBlockElement::GetList(array(), array(), false, array(), $arSelect);?>
Дальше использовать цикл и получить свойства с символьными кодами prop_code_1 и prop_code_2.
Советы веб-разработчиков. Антон Долганин: Если для какого-либо изменения в БД предусмотрен специальный метод, следует использовать именно его, а не более общий метод изменения БД. Хороший пример: модуль интернет-магазина и работа с заказом. Можно изменить флаг оплаты заказа путем CSaleOrder::Update, а можно путем CSaleOrder::PayOrder. Так вот, следует применять именно PayOrder, потому что в нем произойдет вызов соответствующих обработчиков. |
Задача 3:
Добавить свойство типа TEXT/html для элемента.
Если свойство не множественное:
01 <? $element = new CIBlockElement; 02 $PROP = array(); 03 $PROP['символьный код свойства']['VALUE']['TYPE'] = 'text'; // или html 04 $PROP['символьный код свойства']['VALUE']['TEXT'] = 'значение, которое нужно забить'; 05 $arLoadArray = array( 06 "IBLOCK_ID" => IBLOCK_ID, 07 "PROPERTY_VALUES"=> $PROP, 08 "NAME" => "Название элемента" 09 ); 10 $element->Add($arLoadArray);?>
Если свойство множественное:
01 <? // В $ITEMS хранятся значения множественного свойства, которое нужно забить 02 foreach($ITEMS as $item) 03 { 04 $VALUES[]['VALUE'] = array( 05 'TYPE' => 'text', // или html 06 'TEXT' => $item, 07 ); 08 } 09 $element = new CIBlockElement; 10 $PROPS = array(); 11 $PROPS['символьный код свойства'] = $VALUES; 12 $arLoadArray = array( 13 "IBLOCK_ID" => IBLOCK_ID, 14 "PROPERTY_VALUES"=> $PROPS, 15 "NAME" => "Название элемента" 16 ); 17 $element->Add($arLoadArray);?>
Заполнить множественное свойство типа Файл. Довольно часто при добавлении элемента в инфоблок может понадобиться привязать к нему несколько файлов. Для этого удобно создать у инфоблока множественное свойство типа Файл и хранить файлы в нём. Пример заполнения свойства:
01 <? 02 $arFiles = array(); 03 for($i = 1; $i CFile::MakeFileArray($_SERVER["DOCUMENT_ROOT"].'/images/image_'.$i.'.jpg'), 'DESCRIPTION' => ''); 08 } 09 } 10 ?>
После этого массив $arFiles
передается как значение свойства при добавлении элемента.
Задача 5:
Заполнить множественное свойство типа Список с отображением в виде флажков. В данном случае у каждого элемента списка значений есть свой ID. Посмотреть их можно, зайдя в детальное редактирование свойства. Заполняется свойство следующим образом:
1 <? 2 if($first_condition == true) $values[] = array('VALUE' => 1); 3 if($second_condition == true) $values[] = array('VALUE' => 2); 4 CIBlockElement::SetPropertyValuesEx($ELEMENT_ID, $IBLOCK_ID, array('property_code' => $values)); 5 ?>
В данном случае при выполнении первого и второго условий мы отмечаем флажками элементы списка с ID =1 и ID=2 соответственно. Заменить следует $ELEMENT_ID
, $IBLOCK_ID
и property_code
на нужные значения.
Задача 6:
Получить пользовательское свойство раздела
1 <? $section_props = CIBlockSection::GetList(array(), array('IBLOCK_ID' => IBLOCK_ID, 'ID' => SECTION_ID), true, array("UF_ADDITIONAL_PRICE")); 2 $props_array = $section_props->GetNext(); ?>
Теперь в $props_array['UF_ADDITIONAL_PRICE']
лежит значение свойства UF_ADDITIONAL_PRICE раздела инфоблока.
Совет от веб-разработчиков. Алексей Коваленко: При работе с инфоблоками удобнее все коды свойств именовать заглавными буквами. В таком случае вы сможете избежать небольших несостыковок в своей работе. Например, значение свойства с кодом foto при работе с компонентами часто доступно через |
Пример создания своего типа данных для пользовательского свойства
В качестве значения свойства попробуем завести картинку с превью. Это могут быть например фотографии гостиницы на туристическом сайте или что-то подобное. В варианте такого применения и рассмотрим решение задачи.
Один из вариантов реализации: хранить изображения в отдельном инфоблоке и показывать как привязку к элементу. Пример кода:
AddEventHandler("iblock", "OnIBlockPropertyBuildList", array("CIBlockPropertyPicture", "GetUserTypeDescription")); AddEventHandler("iblock", "OnBeforeIBlockElementDelete", array("CIBlockPropertyPicture", "OnBeforeIBlockElementDelete")); class CIBlockPropertyPicture { function GetUserTypeDescription() { return array( "PROPERTY_TYPE" =>"E", "USER_TYPE" =>"Picture", "DESCRIPTION" =>"Картинка", "GetPropertyFieldHtml" =>array("CIBlockPropertyPicture", "GetPropertyFieldHtml"), "GetPublicViewHTML" =>array("CIBlockPropertyPicture", "GetPublicViewHTML"), "ConvertToDB" =>array("CIBlockPropertyPicture", "ConvertToDB"), //"GetPublicEditHTML" =>array("CIBlockPropertyPicture","GetPublicEditHTML"), //"GetAdminListViewHTML" =>array("CIBlockPropertyPicture","GetAdminListViewHTML"), //"CheckFields" =>array("CIBlockPropertyPicture","CheckFields"), //"ConvertFromDB" =>array("CIBlockPropertyPicture","ConvertFromDB"), //"GetLength" =>array("CIBlockPropertyPicture","GetLength"), ); } function GetPropertyFieldHtml($arProperty, $value, $strHTMLControlName) { $LINK_IBLOCK_ID = intval($arProperty["LINK_IBLOCK_ID"]); if($LINK_IBLOCK_ID) { $ELEMENT_ID = intval($value["VALUE"]); if($ELEMENT_ID) { $rsElement = CIBlockElement::GetList(array(), array("IBLOCK_ID" => $arProperty["LINK_IBLOCK_ID"], "ID" => $value["VALUE"]), false, false, array("ID", "PREVIEW_PICTURE", "DETAIL_PICTURE")); $arElement = $rsElement->Fetch(); if(is_array($arElement)) $file_id = $arElement["DETAIL_PICTURE"]; else $file_id = 0; } else { $file_id = 0; } if($file_id) { $db_img = CFile::GetByID($file_id); $db_img_arr = $db_img->Fetch(); if($db_img_arr) { $strImageStorePath = COption::GetOptionString("main", "upload_dir", "upload"); $sImagePath = "/".$strImageStorePath."/".$db_img_arr["SUBDIR"]."/".$db_img_arr["FILE_NAME"]; return '<label><input name="'.$strHTMLControlName["VALUE"].'[del]" value="Y" type="checkbox"> Удалить файл '.$sImagePath.'</label>' .'<input name="'.$strHTMLControlName["VALUE"].'[old]" value="'.$ELEMENT_ID.'" type="hidden">'; } } return '<input type="file" size="'.$arProperty["COL_COUNT"].'" name="'.$strHTMLControlName["VALUE"].'"/>'; } else { return "Ошибка настройки свойства. Укажите инфоблок в котором будут храниться картинки."; } } function GetPublicViewHTML($arProperty, $value, $strHTMLControlName) { $LINK_IBLOCK_ID = intval($arProperty["LINK_IBLOCK_ID"]); if($LINK_IBLOCK_ID) { $ELEMENT_ID = intval($value["VALUE"]); if($ELEMENT_ID) { $rsElement = CIBlockElement::GetList(array(), array("IBLOCK_ID" => $arProperty["LINK_IBLOCK_ID"], "ID" => $value["VALUE"]), false, false, array("ID", "PREVIEW_PICTURE", "DETAIL_PICTURE")); $arElement = $rsElement->Fetch(); if(is_array($arElement)) return CFile::Show2Images($arElement["PREVIEW_PICTURE"], $arElement["DETAIL_PICTURE"]); } } return ""; } function ConvertToDB($arProperty, $value) { $arResult = array("VALUE" => "", "DESCRIPTION" => ""); $LINK_IBLOCK_ID = intval($arProperty["LINK_IBLOCK_ID"]); if($LINK_IBLOCK_ID) { if( is_array($value["VALUE"]) && is_array($value["VALUE"]["error"]) && $value["VALUE"]["error"]["VALUE"] == 0 && $value["VALUE"]["size"]["VALUE"] > 0 ) { $arDetailPicture = array( "name" => $value["VALUE"]["name"]["VALUE"], "type" => $value["VALUE"]["type"]["VALUE"], "tmp_name" => $value["VALUE"]["tmp_name"]["VALUE"], "error" => $value["VALUE"]["error"]["VALUE"], "size" => $value["VALUE"]["size"]["VALUE"], ); $obElement = new CIBlockElement; $arResult["VALUE"] = $obElement->Add(array( "IBLOCK_ID" => $LINK_IBLOCK_ID, "NAME" => $arDetailPicture["name"], "DETAIL_PICTURE" => $arDetailPicture, ), false, false, true); } elseif( is_array($value["VALUE"]) && isset($value["VALUE"]["size"]) && !is_array($value["VALUE"]["size"]) && $value["VALUE"]["size"] > 0 ) { $arDetailPicture = array( "name" => $value["VALUE"]["name"], "type" => $value["VALUE"]["type"], "tmp_name" => $value["VALUE"]["tmp_name"], "error" => intval($value["VALUE"]["error"]), "size" => $value["VALUE"]["size"], ); $obElement = new CIBlockElement; $arResult["VALUE"] = $obElement->Add(array( "IBLOCK_ID" => $LINK_IBLOCK_ID, "NAME" => $arDetailPicture["name"], "DETAIL_PICTURE" => $arDetailPicture, ), false, false, true); } elseif($value["VALUE"]["del"]) { $obElement = new CIBlockElement; $obElement->Delete($value["VALUE"]["old"]); } elseif($value["VALUE"]["old"]) { $arResult["VALUE"] = $value["VALUE"]["old"]; } elseif(!is_array($value["VALUE"]) && intval($value["VALUE"])) { $arResult["VALUE"] = $value["VALUE"]; } } return $arResult; } function OnBeforeIBlockElementDelete($ELEMENT_ID) { $arProperties = array(); $rsElement = CIBlockElement::GetList(array(), array("ID" => $ELEMENT_ID), false, false, array("ID", "IBLOCK_ID")); $arElement = $rsElement->Fetch(); if($arElement) { $rsProperties = CIBlockProperty::GetList(array(), array("IBLOCK_ID" => $arElement["IBLOCK_ID"], "USER_TYPE" => "Picture")); while($arProperty = $rsProperties->Fetch()) $arProperties[] = $arProperty; } $arElements = array(); foreach($arProperties as $arProperty) { $rsPropValues = CIBlockElement::GetProperty($arElement["IBLOCK_ID"], $arElement["ID"], array(), array( "EMPTY" => "N", "ID" => $arProperty["ID"], )); while($arPropValue = $rsPropValues->Fetch()) { $ID = intval($arPropValue["VALUE"]); if($ID > 0) $arElements[$ID] = $ID; } } foreach($arElements as $to_delete) { CIBlockElement::Delete($to_delete); } } }
Что мы в итоге имеем:
- Интерфейс редактирования элемента с возможностью добавления и удаления изображений.
- При удалении элемента связанная с ним информация удаляется.
- Поддержка компонент публичной части.
Инструкция по применению:
- Этот код разместите в файле
/bitrix/php_interface/init.php
. - Создайте инфоблок для хранения изображений и в его настройках укажите параметры генерации картинки предварительного просмотра из детальной (на вкладке Поля).
- В инфоблоке Гостиницы добавьте свойство типа Картинка и в дополнительных настройках этого свойства укажите созданный на первом шаге инфоблок. Не забудьте указать символьный код свойства.
- Создайте элемент и "поиграйтесь" со значениями этого свойства.
- В публичной части, например в компоненте news, в параметрах настройки списка элементов выбрать это свойство.
Как удалить файл в свойстве элемента инфоблока
Обновить любое свойство можно с помощью методов:
При использовании любого метода на ключ массива обновления идет код свойства, а значение - новое значение. Для удаления файла нам надо передать вот такой простой массив:
array('MY_FILE' => array('XXX' => array('del' => 'Y')));
Способ универсален и подходит что для инфоблоков, что для инфоблоков 2.0, что для документооборота. MY_FILE
- это код вашего свойства типа Файл. А что такое ХХХ? В нём содержится ID значения_ свойства. То есть не ID свойства, а именно ID значения.
CModule::IncludeModule('iblock'); $IB = 24; $ID = 220304; $CODE = 'ONE_FL'; if ($arProp = CIBlockElement::GetProperty($IB, $ID, 'ID', 'DESC', array('CODE' => $CODE))->fetch()) { $XXX = $arProp['PROPERTY_VALUE_ID']; CIBlockElement::SetPropertyValueCode($ID, $CODE, array($XXX => array('del' => 'Y'))); }
Вот таким образом получается этот универсальный XXX, именно так его и надо передавать для каждого файла, который мы хотим удалить.
Что делать в случае множественного файла? Как удалить конкретный файл в списке? Все просто - используем в примере выше не if, а while, ну и дополнительно фильтруем, какой файл надо удалить.
Назад в раздел