Вывод новостей сразу в нескольких новостных лентах — различия между версиями
Mad grant (обсуждение | вклад) |
Mad grant (обсуждение | вклад) |
||
Строка 41: | Строка 41: | ||
<source lang="php"> | <source lang="php"> | ||
public function customNewsList($elementPath, $template = "default", $per_page = false, $ignore_paging = false){ | public function customNewsList($elementPath, $template = "default", $per_page = false, $ignore_paging = false){ | ||
+ | //если $per_page не задан, то берем его из настроек модуля Новости | ||
if(!$per_page){ | if(!$per_page){ | ||
(int) $per_page = $this->per_page; | (int) $per_page = $this->per_page; | ||
} | } | ||
+ | //загружаем шаблон | ||
list($template_block, $template_block_empty, $template_line) = def_module::loadTemplates("news/".$template, "items_block", "items_block_empty", "item_block"); | list($template_block, $template_block_empty, $template_line) = def_module::loadTemplates("news/".$template, "items_block", "items_block_empty", "item_block"); | ||
+ | //берем параметр 'p' | ||
$currPageNum = (int) getRequest('p'); | $currPageNum = (int) getRequest('p'); | ||
+ | //если в $elementPath указан путь до ленты, то вытаскиваем её id | ||
$parentId = $this->analyzeRequiredPath($elementPath); | $parentId = $this->analyzeRequiredPath($elementPath); | ||
− | + | //если такой страницы нет, то кидаем исключение | |
if($parentId === false){ | if($parentId === false){ | ||
throw new publicException(getLabel('error-page-does-not-exist', null, $elementPath)); | throw new publicException(getLabel('error-page-does-not-exist', null, $elementPath)); | ||
} | } | ||
+ | //задаем массив $itemsFromCurrLent, куда попадут новости из выбранной ленты | ||
$itemsFromCurrLent = array(); | $itemsFromCurrLent = array(); | ||
+ | //получаем эти новости: | ||
$currLentItems = new selector('pages'); | $currLentItems = new selector('pages'); | ||
$currLentItems->types('object-type')->name('news', 'item'); | $currLentItems->types('object-type')->name('news', 'item'); | ||
$currLentItems->where('hierarchy')->page($parentId); | $currLentItems->where('hierarchy')->page($parentId); | ||
+ | //записываем их в массив $itemsFromCurrLent | ||
$itemsFromCurrLent = $currLentItems->result(); | $itemsFromCurrLent = $currLentItems->result(); | ||
unset($currLentItems); | unset($currLentItems); | ||
− | + | //задаем массив $itemsWithCurrLink, куда попадут новости из всех лент, у которых задана выбранная лента | |
$itemsWithCurrLink = array(); | $itemsWithCurrLink = array(); | ||
+ | //получаем эти новости | ||
$allLentsItems = new selector('pages'); | $allLentsItems = new selector('pages'); | ||
$allLentsItems->types('object-type')->name('news', 'item'); | $allLentsItems->types('object-type')->name('news', 'item'); | ||
$allLentsItems->where('rubric_link')->equals($parentId); | $allLentsItems->where('rubric_link')->equals($parentId); | ||
+ | //записываем их в массив $itemsWithCurrLink | ||
$itemsWithCurrLink = $allLentsItems->result(); | $itemsWithCurrLink = $allLentsItems->result(); | ||
unset($allLentsItems); | unset($allLentsItems); | ||
− | + | //задаем результирующий массив новостей | |
$resultItems = array(); | $resultItems = array(); | ||
+ | //помещаем в него все элементы из массива $itemsFromCurrLent и $itemsWithCurrLink, удаляем дубли. | ||
$resultItems = array_unique(array_merge($itemsFromCurrLent, $itemsWithCurrLink)); | $resultItems = array_unique(array_merge($itemsFromCurrLent, $itemsWithCurrLink)); | ||
Версия 13:42, 23 октября 2013
Актуально для версии 2.9.1
Содержание
Задача
Иногда необходимо, чтобы одна и та же новость отображалась в нескольких лентах новостей, стандартными средствами можно создавать виртуальные или просто копии новости и перетаскивать их через модуль "Структура" по нужным разделам, а если на сайте сложная структура новостных лент, это не совсем удобно.
Поэтому, нам нужно реализовать функционал, который позволит указывать на странице конкретной новости, в каком разделе её выводить.
Решение
Со стороны административной панели, нам понадобиться создать в типе данных 'Новость' поле типа "ссылка на дерево" с идентификатором 'rubric_link', все эти действия производятся в модуле "Шаблоны данных", в результате мы получим примерно следующее:
Теперь мы можем у каждой новости указать раздел для вывода:
Разделы мы указали, теперь нам понадобиться кастомный макрос, который будет выводить новости из конкретной ленты новостей, плюс новости из всех лент, у которых в поле с идентификатором 'rubric_link' выбрана эта лента новостей. Код данного макроса и его описание даны ниже.
Описание макроса
%news customNewsList()% — выводит новости из выбранной ленты и новости из всех лент новостей, у которых в поле типа "ссылка на дерево" с идентификатором 'rubric_link' указана выбранная лента.
Параметры: customNewsList($elementPath,[$template = "default", $per_page = false, $ignore_paging = false])
$elementPath
Принимает id или путь до ленты новостей, новости из которой будут выводиться макроса
$template
Принимает имя шаблона, по которому выводится результат макроса. В XSLT-шаблонизаторе игнорируется.
$per_page
Принимает число, которое обозначает максимальное количество новостей. Если этот параметр не указывать, будет взято значение, указанное в настройках модуля «Новости».
$ignore_paging
Принимаем булево значение («1» или «0»), указывающее макросу игнорировать значение текущей страницы списка вывода (параметр http-запроса p). То есть, если указать «1», макрос будет всегда выводить только первую страницу списка новостей
Для применения этого макроса скопируйте содержимое листинга макроса в файл /classes/modules/news/__custom.php, и не забудьте указать имя метода в permissions.custom.php.
Листинг макроса
public function customNewsList($elementPath, $template = "default", $per_page = false, $ignore_paging = false){
//если $per_page не задан, то берем его из настроек модуля Новости
if(!$per_page){
(int) $per_page = $this->per_page;
}
//загружаем шаблон
list($template_block, $template_block_empty, $template_line) = def_module::loadTemplates("news/".$template, "items_block", "items_block_empty", "item_block");
//берем параметр 'p'
$currPageNum = (int) getRequest('p');
//если в $elementPath указан путь до ленты, то вытаскиваем её id
$parentId = $this->analyzeRequiredPath($elementPath);
//если такой страницы нет, то кидаем исключение
if($parentId === false){
throw new publicException(getLabel('error-page-does-not-exist', null, $elementPath));
}
//задаем массив $itemsFromCurrLent, куда попадут новости из выбранной ленты
$itemsFromCurrLent = array();
//получаем эти новости:
$currLentItems = new selector('pages');
$currLentItems->types('object-type')->name('news', 'item');
$currLentItems->where('hierarchy')->page($parentId);
//записываем их в массив $itemsFromCurrLent
$itemsFromCurrLent = $currLentItems->result();
unset($currLentItems);
//задаем массив $itemsWithCurrLink, куда попадут новости из всех лент, у которых задана выбранная лента
$itemsWithCurrLink = array();
//получаем эти новости
$allLentsItems = new selector('pages');
$allLentsItems->types('object-type')->name('news', 'item');
$allLentsItems->where('rubric_link')->equals($parentId);
//записываем их в массив $itemsWithCurrLink
$itemsWithCurrLink = $allLentsItems->result();
unset($allLentsItems);
//задаем результирующий массив новостей
$resultItems = array();
//помещаем в него все элементы из массива $itemsFromCurrLent и $itemsWithCurrLink, удаляем дубли.
$resultItems = array_unique(array_merge($itemsFromCurrLent, $itemsWithCurrLink));
(int) $total = count($resultItems);
if($total>0){
$limitedItems = array();
if($ignore_paging){
$limitedItems = $resultItems;
}else{
$offset = $currPageNum * $per_page;
$limitedItems = array_slice($resultItems, $offset, $per_page);
}
unset($resultItems);
usort($limitedItems, function($a, $b){
$a_publishTime = $a->getValue('publish_time');
$b_publishTime = $b->getValue('publish_time');
if ($a_publishTime == $b_publishTime) {
return 0;
}
return ($a_publishTime > $b_publishTime) ? -1 : 1;
});
$lines = Array();
$block_arr = Array();
foreach ($limitedItems as $item){
$line_arr = array();
$itemId = $item->id;
$line_arr['attribute:id'] = $item->id;
$line_arr['attribute:link'] = $item->link;
$line_arr['xlink:href'] = "upage://" . $itemId;
$line_arr['void:header'] = $lines_arr['name'] = $item->getName();
$line_arr['node:name'] = $item->getName();
if($publish_time = $item->getValue('publish_time')){
$line_arr['attribute:publish_time'] = $publish_time->getFormattedDate("U");
}
unset($item);
$lines[] = def_module::parseTemplate($template_line, $line_arr, $itemId);
unset($line_arr);
$this->pushEditable("news", "rubric", $itemId);
unset($itemId);
}
if(is_array($parentId)) {
list($parentId) = $parentId;
}
$block_arr['subnodes:items'] = $block_arr['void:lines'] = $lines;
unset($lines);
$block_arr['total'] = $total;
$block_arr['per_page'] = $per_page;
return def_module::parseTemplate($template_block, $block_arr, $parentId);
}else{
return $template_block_empty;
}
}