Новости Joomla

Как триггерить события для плагинов на манер Joomla 5+?В Joomla 6 должны удалить метод...

Как триггерить события для плагинов на манер Joomla 5+?В Joomla 6 должны удалить метод...

👩‍💻 Как триггерить события для плагинов на манер Joomla 5+?В Joomla 6 должны удалить метод triggerEvent(), с помощью которого раньше вызывались события для плагинов. Теперь чтобы в своём коде вызвать событие для плагина и получить от него результаты нужно:- создать объект класса события- передать в него параметры

use Joomla\CMS\Event\AbstractEvent;use Joomla\CMS\Factory;use Joomla\CMS\Plugin\PluginHelper;// Грузим плагины нужных группPluginHelper::importPlugin('system');// Создаём объект события$event = AbstractEvent::create('onAfterInitUniverse', [    'subject' => $this,    'data'    => $data, // какие-то данные    'article' => $article, // ещё материал вдовесок    'product' => $product, // и товаров подвезли]);// Триггерим событиеFactory::getApplication()->getDispatcher()->dispatch(    $event->getName(), // Тут можно строку передать 'onAfterInitUniverse'    $event);// Получаем результаты// В случае с AbstractEvent это может быть не 'result',// а что-то ещё - куда сами отдадите данные.// 2-й аргумент - значение по умолчанию, // если не получены результаты$results = $event->getArgument('result', []);
Плюсы такого подхода - вам не нужно запоминать порядок аргументов и проверять их наличие. Если вы написали свой класс события, то в плагине можно получать аргументы с помощью методов $event->getArticle(), $event->getData(), $event->getProduct() и подобными - реализуете сами под свои нужды. Если такой класс события написали, то создаёте экземпляр своего класса события и укажите его явно в аргументе eventClass
use Joomla\Component\MyComponent\Administrator\Event\MyCoolEvent;$event = MyCoolEvent::create('onAfterInitUniverse', [    'subject'    => $this,    'eventClass' => MyCoolEvent::class, // ваш класс события    'data'       => $data, // какие-то данные    'article'    => $article, // ещё материал вдовесок    'product'    => $product, // и товаров подвезли]);
Ожидаемо, что класс вашего события будет расширять AbsractEvent или другие классы событий Joomla.🙁 Есть неприятный нюанс - нельзя просто так вызывать событие и ничего не передать в аргументы. Аргумент subject обязательный. Но если вы всё-таки не хотите туда ничего передавать - передайте туда пустой stdClass или объект Joomla\registry\Registry.
@joomlafeed#joomla #php #webdev

0 Пользователей и 1 Гость просматривают эту тему.
  • 5 Ответов
  • 2146 Просмотров
*

devirus

  • Захожу иногда
  • 140
  • 6 / 0
Продолжаю пилить, застрял на дополнительных БД запросах.

Задача: выводить сопутствующие товары к каждому item на странице категории.

Откопал tables/category.php с методом getProducts(), которое обращается к lib/jtableauto.php (для построения запроса)
Есть событие onBeforeQueryGetProductList которое использую в плагине (может и не оно нужно?)
Есть переменные как раз для дополнительных строк в запросе, как я понял.
И есть в tables/product.php метод getRelatedProducts(), но он только для страницы товара.

В общем пока не сообразил как собрать всё вместе..

upd: начинаю тихонько вникать, только загвоздка на уровне знания php:
в классе C есть метод M, который вызывает другой метод L этого же класса C вот так: $this->L ...НО когда я вызываю метод M в классе X вот так: C::M то эта самая $this получается в классе X, в котором, конечно нет метода L — получается ошибка. (могу менять только класс X, то есть плагин)

upd2: так, мне не нужно всё это, думаю достаточно грамотно составить sql-запрос, т. к. плагином можно добавить $adv_query и $adv_from прямо в запрос... еще бы понимать синтаксис sql =)
« Последнее редактирование: 18.06.2014, 14:48:16 от devirus »
*

devirus

  • Захожу иногда
  • 140
  • 6 / 0
Решено :)

Код плагина, может пригодится кому.
Добавляет в стандартный $this->rows ключи "rel" с сопутствующими id для каждого товара на странице. А сами сопутствующие товары выгружаются в $this->related
Остается перебрать и построить нужный шаблон...

Код
class plgJshoppingProductsStarkman_relatedInCategory extends JPlugin{

    function onBeforeQueryGetProductList($type, &$adv_result, &$adv_from, &$adv_query, &$order_query, &$filters){

    if ($type == "category"){
    $adv_result .= ", GROUP_CONCAT(relation.product_related_id) as rel";
        $adv_from .= "LEFT JOIN `#__jshopping_products_relations` AS relation ON relation.product_id = prod.product_id";
        $adv_query .= " group by prod.product_id";
    }

    }

    function onBeforeDisplayProductListView(&$view){

        foreach ($view->rows as $k => $product) {
        $k==0 ? $products_in_page = $product->product_id : $products_in_page .= ',' . $product->product_id;
        }

        $adv_query = "";
        $adv_from = "";
        $adv_result = JTableAvto::getBuildQueryListProductDefaultResult();
        $filters = array();

        $jshopConfig = JSFactory::getConfig();
        $user = JFactory::getUser();

        $groups = implode(',', $user->getAuthorisedViewLevels());
        $adv_query .=' AND prod.access IN ('.$groups.') AND cat.access IN ('.$groups.')';
       
        if ($jshopConfig->hide_product_not_avaible_stock){
            $adv_query .= " AND prod.product_quantity > 0";
        }
        if ($jshopConfig->show_delivery_time){
            $adv_result .= ", prod.delivery_times_id";
        }
        if ($jshopConfig->admin_show_product_extra_field){
            $adv_result .= getQueryListProductsExtraFields();
        }
        if ($jshopConfig->product_list_show_vendor){
            $adv_result .= ", prod.vendor_id";
        }
        if ($jshopConfig->product_list_show_qty_stock){
            $adv_result .= ", prod.unlimited";
        }

        $order_query = "";

        $query = "SELECT $adv_result FROM `#__jshopping_products_relations` AS relation
                INNER JOIN `#__jshopping_products` AS prod ON relation.product_related_id = prod.product_id
                LEFT JOIN `#__jshopping_products_to_categories` AS pr_cat ON pr_cat.product_id = relation.product_related_id
                LEFT JOIN `#__jshopping_categories` AS cat ON pr_cat.category_id = cat.category_id
                $adv_from
                WHERE relation.product_id IN (" . $products_in_page . ") AND cat.category_publish='1' AND prod.product_publish = '1' ".$adv_query." group by prod.product_id ".$order_query;
       

        $db = &JFactory::getDBO();
$db->setQuery($query);
$rel = $db->loadObjectList();
       
        foreach($rel as $key=>$value) {
            $rel[$key]->product_link = SEFLink('index.php?option=com_jshopping&controller=product&task=view&category_id='.$value->category_id.'&product_id='.$value->product_id, 1);
        }
        $rel = listProductUpdateData($rel, 1);

        $view->assign('related', $rel);
    }
}
*

dmitry_stas

  • Легенда
  • 13151
  • 1234 / 8
1. Если все равно
Цитировать
Остается перебрать и построить...
то может в принципе исключить onBeforeQueryGetProductList, и чуть изменить запрос выборки сопутствующих ? Можно конечно и так, но я всегда аккуратничаю с добавлением в плагине в запрос group by, потому что если какой нибудь другой плагин добавит например еще один leftjoin, то фатала не будет, но если будет несколько group by...

2. Чтобы вызывать так JTableAvto::getBuildQueryListProductDefaultResult(), метод должен быть объявлен как статический. Ошибки нет, но warning есть.

3. Может грохнуть простынку с определением $adv_query и $adv_result? Ведь у JTableAvto есть метод getBuildQueryListProduct

4. В listProductUpdateData второй параметр определяет, будут ли добавлены ссылки для товаров. Соответственно, перед вызовом лишний цикл.
Тут дарят бакс просто за регистрацию! Успей получить!
Все советы на форуме раздаю бесплатно, то есть даром. Индивидуально бесплатно консультирую только по вопросам стоимости индивидуальных консультаций
*

devirus

  • Захожу иногда
  • 140
  • 6 / 0
Писал методом тыка..) с комментариями согласен, спасибо!

1) чуть изменить — в смысле оставить в покое $this->rows и с другого конца подойти: в $this->related сделать привязку к товарам? Там что-то у меня Неправильно вроде сгруппировалось... попробую еще поковырять.
2) а как нужно? делал new JTableAvto - ошибки сыпятся
3) на метод getBuildQueryListProduct смотрел, но там в конце есть $this->getBuildQueryListProductFilterPrice($filters, $adv_query, $adv_from); когда вызываю JTableAvto::getBuildQueryListProduct, то $this обрабатывается внутри плагина и не находит метод (не хватает мозгов как обойти, видимо тот же new JTableAvto неправильно пишу).
4) listProductUpdateData не знал для чего, остался от продуктовой getRelatedProducts... уберу параметр
*

dmitry_stas

  • Легенда
  • 13151
  • 1234 / 8
1. угу, в $this->related сделать привязку к товарам
2. JTableAvto - это расширение JTable. Соответственно, получаем синглтон абсолютно любой таблицы ЖШ, которая является расширением JTableAvto, и дальше уже работаем как с таблицей:
Код
$tProduct = JTable::getInstance('Product', 'jshop');
$adv_result = $tProduct->getBuildQueryListProductDefaultResult()
3. см п.2
4. ну или так :)
Тут дарят бакс просто за регистрацию! Успей получить!
Все советы на форуме раздаю бесплатно, то есть даром. Индивидуально бесплатно консультирую только по вопросам стоимости индивидуальных консультаций
*

devirus

  • Захожу иногда
  • 140
  • 6 / 0
Цитировать
перед вызовом лишний цикл
...дошло)

доработанный код
Код
    function onBeforeDisplayProductListView(&$view){

        foreach ($view->rows as $k => $product) {
        $k==0 ? $products_in_page = $product->product_id : $products_in_page .= ',' . $product->product_id;
        }

        $adv_query = "";
        $adv_from = "";
        $filters = array();
        $order_query = "";

        $tProduct = JTable::getInstance('Product', 'jshop');
$adv_result = $tProduct->getBuildQueryListProductDefaultResult();
        $tProduct->getBuildQueryListProductSimpleList("related", null, $filters, $adv_query, $adv_from, $adv_result);

        $adv_result .= ", GROUP_CONCAT(relation.product_id) as rel";

        $query = "SELECT $adv_result FROM `#__jshopping_products_relations` AS relation
                INNER JOIN `#__jshopping_products` AS prod ON relation.product_related_id = prod.product_id
                LEFT JOIN `#__jshopping_products_to_categories` AS pr_cat ON pr_cat.product_id = relation.product_related_id
                LEFT JOIN `#__jshopping_categories` AS cat ON pr_cat.category_id = cat.category_id
                $adv_from
                WHERE relation.product_id IN (" . $products_in_page . ") AND cat.category_publish='1' AND prod.product_publish = '1' ".$adv_query." group by prod.product_id ".$order_query;

        $db = &JFactory::getDBO();
$db->setQuery($query);
$rel = $db->loadObjectList();
       
        $rel = listProductUpdateData($rel, 1);

        $view->assign('related', $rel);
    }
Чтобы оставить сообщение,
Вам необходимо Войти или Зарегистрироваться
 

[Решено] - Заказ на одной странице

Автор shurakana

Ответов: 58
Просмотров: 18990
Последний ответ 16.03.2023, 09:53:16
от kit2m2
Плагин - Количество товаров в категории [Скачать]

Автор kit2m2

Ответов: 8
Просмотров: 2782
Последний ответ 06.07.2021, 18:31:46
от kit2m2
Таблица размеров (и т.п.) Плагин модульных позиций для атрибутов на странице товара [Скачать]

Автор kit2m2

Ответов: 39
Просмотров: 7991
Последний ответ 16.05.2021, 18:16:50
от kit2m2
Прописать rel canonical в товары с атрибутами в url

Автор Unclesem

Ответов: 4
Просмотров: 658
Последний ответ 24.02.2021, 20:50:13
от Unclesem
Плагин для покупки в категории товара JoomShopping 4, с применением ajax

Автор draff

Ответов: 14
Просмотров: 1274
Последний ответ 18.05.2020, 06:29:43
от draff