Женим mFilter2 и tvSuperSelect
Понадобилось однажды сделать фильтры по диапазону. То есть, например, в поселке Один участки площадью 10-15 соток, а в поселке Два — 13-20 соток. Фильтровать нужно поселки по по площади участка. В итоге я пришел к тому, чтобы в tvSuperSelectпоселку писать крайние значения (10, 15), а на фронтенде — стандартные ui-slider от mFilter2…
И тут выяснилось, что mFilter не знаком с суперселектом)
Поковырялся и сделал следующее:
1. Создаем по инструкции от Василия расширяющий класс для фильтра. Я его обозвал SuperSelectFilter. В нем переопределяем функцию getTvValues. Код почти весь оставляем прежним, только там, где создается переменная $tmp (массив значений для фильтра) добавляем условие для типа tvSuperSelect.
Работает без нареканий. Ура, товарищи.
P. S. Если есть какие другие идеи, как подобную фильтрацию сделать — делитесь…
И тут выяснилось, что mFilter не знаком с суперселектом)
Поковырялся и сделал следующее:
1. Создаем по инструкции от Василия расширяющий класс для фильтра. Я его обозвал SuperSelectFilter. В нем переопределяем функцию getTvValues. Код почти весь оставляем прежним, только там, где создается переменная $tmp (массив значений для фильтра) добавляем условие для типа tvSuperSelect.
if ($row['type'] == 'tvSuperSelect') {
$tmp = json_decode($row['value']);
} else {
$tmp = strpos($row['value'], '||') !== false ? explode('||', $row['value']) : array($row['value']);
}
В принципе, этот же костыль должен сработать с любым json-значением, например migx, но я не пробовал. Вот получившийся класс:<?php
class SuperSelectFilter extends mse2FiltersHandler {
public function getTvValues(array $tvs, array $ids) {
$filters = array();
$q = $this->modx->newQuery('modResource', array('modResource.id:IN' => $ids));
$q->leftJoin('modTemplateVarTemplate', 'TemplateVarTemplate',
'TemplateVarTemplate.tmplvarid IN (SELECT id FROM ' . $this->modx->getTableName('modTemplateVar') . ' WHERE name IN ("' . implode('","', $tvs) . '") )
AND modResource.template = TemplateVarTemplate.templateid'
);
$q->leftJoin('modTemplateVar', 'TemplateVar', 'TemplateVarTemplate.tmplvarid = TemplateVar.id');
$q->leftJoin('modTemplateVarResource', 'TemplateVarResource', 'TemplateVarResource.tmplvarid = TemplateVar.id AND TemplateVarResource.contentid = modResource.id');
$q->select('TemplateVar.name, TemplateVarResource.contentid as id, TemplateVarResource.value, TemplateVar.type, TemplateVar.default_text');
$tstart = microtime(true);
if ($q->prepare() && $q->stmt->execute()) {
$this->modx->queryTime += microtime(true) - $tstart;
$this->modx->executedQueries++;
while ($row = $q->stmt->fetch(PDO::FETCH_ASSOC)) {
if (empty($row['id'])) {
continue;
}
elseif (is_null($row['value']) || trim($row['value']) == '') {
$row['value'] = $row['default_text'];
}
if ($row['type'] == 'tag' || $row['type'] == 'autotag') {
$row['value'] = str_replace(',', '||', $row['value']);
}
if ($row['type'] == 'tvSuperSelect') {
$tmp = json_decode($row['value']);
} else {
$tmp = strpos($row['value'], '||') !== false ? explode('||', $row['value']) : array($row['value']);
}
foreach ($tmp as $v) {
$v = str_replace('"', '"', trim($v));
if ($v == '') {
continue;
}
$name = strtolower($row['name']);
if (isset($filters[$name][$v])) {
$filters[$name][$v][$row['id']] = $row['id'];
}
else {
$filters[$name][$v] = array($row['id'] => $row['id']);
}
}
}
}
else {
$this->modx->log(modX::LOG_LEVEL_ERROR, "[mSearch2] Error on get filter params.\nQuery: ".$q->toSQL()."\nResponse: ".print_r($q->stmt->errorInfo(),1));
}
return $filters;
}
}
2. Переопределил системную настройку «mse2_filters_handler_class» — вписал туда название своего класса.Работает без нареканий. Ура, товарищи.
P. S. Если есть какие другие идеи, как подобную фильтрацию сделать — делитесь…
Комментарии: 5
А есть живой пример?
есть, angeldela.ru
не, ну Ваще круто работает!
А сам вывод mfilter не подскажете?
Огромное спасибо!
Авторизуйтесь или зарегистрируйтесь, чтобы оставлять комментарии.