Видаліть блок із макета без імені


12

Я хочу видалити блок з макета в magento 2, який оголошується в розширенні сторонніх розробників, але в блоці немає імені.
Чи можу я це зробити?

Блок оголошується так

<referenceContainer name="before.body.end">
    <block class="Magento\Backend\Block\Template" template="[Vendor_Module]::template.phtml"/>
</referenceContainer>

Я не можу використовувати

<referenceBlock name="..." remove="true" /> 

бо, як бачите, на ньому немає імені.


Маріус, у мене є ідея. Якщо ми використовуємо подію та видалимо блок за назвою шаблону відповідності [Vendor_Module]::template.phtml
Amit Bera

У мене така ж ідея (див. Коментарі до відповіді), але я буду використовувати її лише як відчайдушний захід. Я сподівався на просте рішення. Якщо у вас є якийсь код, опублікуйте його як відповідь.
Маріус

ха-ха, що у нас немає простого рішення. Попробуй дати відповідь, використовуючи подію
Аміт Бера

Відповіді:


5

Я знайшов це питання на уроці Magento\Framework\View\Layout\ScheduledStructure\Helper

Є функція _generateAnonymousName:

protected function _generateAnonymousName($class)
{
    $position = strpos($class, '\\Block\\');
    $key = $position !== false ? substr($class, $position + 7) : $class;
    $key = strtolower(trim($key, '_'));
    return $key . $this->counter++;
}

Це дзвінок з scheduleStructureфункції:

    public function scheduleStructure(
    Layout\ScheduledStructure $scheduledStructure,
    Layout\Element $currentNode,
    Layout\Element $parentNode
) {
    // if it hasn't a name it must be generated
    if (!(string)$currentNode->getAttribute('name')) {
        $name = $this->_generateAnonymousName($parentNode->getElementName() . '_schedule_block'); // CALL HERE
        $currentNode->setAttribute('name', $name);
    }
    $path = $name = (string)$currentNode->getAttribute('name');

    // Prepare scheduled element with default parameters [type, alias, parentName, siblingName, isAfter]
    $row = [
        self::SCHEDULED_STRUCTURE_INDEX_TYPE           => $currentNode->getName(),
        self::SCHEDULED_STRUCTURE_INDEX_ALIAS          => '',
        self::SCHEDULED_STRUCTURE_INDEX_PARENT_NAME    => '',
        self::SCHEDULED_STRUCTURE_INDEX_SIBLING_NAME   => null,
        self::SCHEDULED_STRUCTURE_INDEX_IS_AFTER       => true,
    ];

    $parentName = $parentNode->getElementName();
    //if this element has a parent element, there must be reset [alias, parentName, siblingName, isAfter]
    if ($parentName) {
        $row[self::SCHEDULED_STRUCTURE_INDEX_ALIAS] = (string)$currentNode->getAttribute('as');
        $row[self::SCHEDULED_STRUCTURE_INDEX_PARENT_NAME] = $parentName;

        list($row[self::SCHEDULED_STRUCTURE_INDEX_SIBLING_NAME],
            $row[self::SCHEDULED_STRUCTURE_INDEX_IS_AFTER]) = $this->_beforeAfterToSibling($currentNode);

        // materialized path for referencing nodes in the plain array of _scheduledStructure
        if ($scheduledStructure->hasPath($parentName)) {
            $path = $scheduledStructure->getPath($parentName) . '/' . $path;
        }
    }

    $this->_overrideElementWorkaround($scheduledStructure, $name, $path);
    $scheduledStructure->setPathElement($name, $path);
    $scheduledStructure->setStructureElement($name, $row);
    return $name;
}

У цьому випадку ім'ям блоку може бути:

  • before.body.end_schedule_block1
  • before.body.end_schedule_block2
  • ...

Я думаю, вам слід визначити блок підсумків без імені на контейнері, а ім'я блоку замовлення потрібно видалити з контейнера.


Я не думаю, що це спрацює. Немає можливості передбачити створене ім'я, оскільки на різних сторінках в body.before.endконтейнері може бути додано кілька блоків у різному порядку.
Маріус

Цей випадок стосується лише блоку / контейнера без імені. Якщо всі вони без імені, так важко визначити якийсь блок / контейнер потрібно видалити.
Тао Фам

так ... моя проблема саме
Маріус

Ми повинні переписати $name = $this->_generateAnonymousName($parentNode->getElementName() . '_schedule_block');, чи слід передати клас та шаблон параметром?
Тао Фам

2
видається накладним, щоб переписати щось подібне. Я шукаю просте рішення (якщо воно є) або відповідь на кшталт "не можливо дуже легко". Я думаю, що я можу спостерігати за компонуванням подій, що генерує блоки подій, або щось, щоб їх видалити, але знову здається, що надто багато накладних витрат. Я зберігаю це як резервне рішення.
Маріус

3

Я дійсно даю вам погану ідею.

Тут ідея не зупинити вихід вашого блоку

Використання події view_block_abstract_to_html_after

<?xml version="1.0"?>
<config xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:noNamespaceSchemaLocation="urn:magento:framework:Event/etc/events.xsd">
    <event name="view_block_abstract_to_html_after">
        <observer name="myObserverName" instance="Stack\Work\Observer\MyObserver" />
    </event>
</config>

І за допомогою цього спостерігача вимкніть вихід блоку

<?php
namespace Stack\Work\Observer;
use Magento\Framework\Event\ObserverInterface;

class MyObserver implements ObserverInterface
{
  public function __construct()
  {
    //Observer initialization code...
    //You can use dependency injection to get any class this observer may need.
  }

  public function execute(\Magento\Framework\Event\Observer $observer)
  {
    $block = $observer->getData('block');

    if('[Vendor_Module]::template.phtml' == $block->getTemplate()){
        $transport = $observer->getData('transport');
        $transport->setHtml('');

    }
  }
}

це насправді не така погана ідея. Дійсно, є надмірне спостереження за всіма блоками, але я готовий використовувати його над іншими варіантами. Я спробую і повідомити вам.
Маріус

охолоджувати. людина .... подивись, що станеться
Аміт Бера

1
Це працює, але я намагався трохи його оптимізувати, а не виконувати код для кожного блоку. Так я закінчив свою відповідь . Дякую за ідею.
Маріус

Я бачу відповідь, що справді хороший чоловік :)
Аміт Бера

3

Я отримав ідею з відповіді Amit і закінчив робоче рішення, яке виглядає не дуже нав'язливо, і це не зайвий набір, оскільки мій код виконується лише один раз.

Я створив спостерігача за подією, layout_generate_blocks_afterяка виконується після завантаження макетів та генерування блоків.

Це може мати недолік, оскільки блок, який я намагаюся видалити, все-таки отримує екземпляр, але в моєму випадку мені просто потрібно було його видалити зі сторінки.

Отже, я маю файл etc/adminhtml/events.xml

<?xml version="1.0"?>
<config xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:noNamespaceSchemaLocation="urn:magento:framework:Event/etc/events.xsd">
    <event name="layout_generate_blocks_after">
        <observer name="remove-the-block" instance="[MyVendor]\[MyModule]\Observer\RemoveBlock" />
    </event>
</config>

і мій клас спостерігача:

<?php
namespace [MyVendor]\[MyModule]\Observer;

use Magento\Framework\Event\Observer;
use Magento\Framework\Event\ObserverInterface;

class RemoveBlock implements ObserverInterface
{
    const TEMPLATE_TO_REMOVE = '[OtherVendor]_[OtherModule]::template.phtml';
    public function execute(Observer $observer)
    {
        /** @var \Magento\Framework\View\Layout $layout */
        $layout = $observer->getLayout();
        $blocks = $layout->getAllBlocks();
        foreach ($blocks as $key => $block) {
            /** @var \Magento\Framework\View\Element\Template $block */
            if ($block->getTemplate() == self::TEMPLATE_TO_REMOVE) {
                $layout->unsetElement($key);
            }
        }
    }
}
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.