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

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
Амит Бера
У меня та же идея (см. Комментарии к ответу), но я буду использовать ее только в качестве отчаянной меры. Я надеялся на простое решение. Если у вас есть код, опубликуйте его как ответ.
Мариус
ха-ха, что у нас нет простого решения. позвольте мне дать вам ответ, используя событие
Амит Бера

Ответы:

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('');

    }
  }
}
Amit Бера
источник
на самом деле это не такая плохая идея. На самом деле, наблюдается перегиб, наблюдающий за всеми блоками, но я готов использовать его поверх других вариантов. Я постараюсь и дам вам знать.
Мариус
Coool. мужик .... посмотри что получится
Амит Бера
1
Это работает, но я попытался немного его оптимизировать, чтобы не выполнять код для каждого блока. В итоге я получил ответ . Спасибо за идею.
Мариус
Я вижу ответ, что действительно хороший человек :)
Амит Бера
3

Я получил идею из ответа Амита и в результате получил рабочее решение, которое не выглядит слишком навязчивым и не является излишним, поскольку мой код выполняется только один раз.

Я создал наблюдателя на событие 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);
            }
        }
    }
}
Мариус
источник