【问题标题】:Override entire magento layout from observer从观察者覆盖整个magento布局
【发布时间】:2013-07-01 19:02:18
【问题描述】:

问题空间

我正在尝试使用 magento 中的观察者根据请求中的参数完全替换给定请求的整个布局。

我面临的问题是 Magento 仍在尝试加载与我在 Observer 中指定的不同的根块模板(特别是产品页面的“frontend/base/default/template/page/1column.phtml”这是后端设计选项卡中配置的模板使用的默认根块模板)。因为它没有使用我指定的布局,所以它在 PHP 中尝试加载主题中不存在的模板时死掉了。

感谢任何指导。

注意:我没有使用 CMS 页面来测试这个概念,因为它们会加载自己在后端指定的模板。我创建了一个测试产品并正在使用它的产品页面。通过请求以下 URL 进行测试:http://mymagentosite.com/test-product?ajax=1

可能的问题

  • 我可能没有听正确的事件来完全替换布局。文档很少,所以我根据 Stack Overflow 对布局问题的其他回复进行猜测。
  • 布局元素的文件夹结构对我来说似乎是巫术,因为我发现关于该主题的不同意见(例如 page.xml 与 local.xml)

实施

我创建了一个带有观察者的模块和一个非常小的布局,如下所示:

模块文件夹结构

app
├── code
│   └── local
│       └── MyCompany
│           └── MyModule
│               ├── etc
│               │   └── config.xml
│               └── Model
│                   └── Observer.php
├── design
│   └── frontend
│       └── myTheme
│           └── default
│               ├── layout
│               │   └── local.xml
│               └── template
│                   └── test.phtml
└── etc
    └── modules
        └── MyCompany_MyModule.xml

config.xml

<!-- language: lang-xml -->
<?xml version="1.0" encoding="UTF-8"?>
<config>
    <modules>
        <MyCompany_MyModule>
            <version>0.1.0.0</version>
        </MyCompany_MyModule>
    </modules>
    <global>
        <events>
            <controller_action_layout_generate_xml_before>
                <observers>
                    <myCompany_myModule_model_observer>
                        <type>singleton</type>
                        <class>MyCompany_MyModule_Model_Observer</class>
                        <method>changeRequestLayout</method>
                    </myCompany_myModule_model_observer>
                </observers>
            </controller_action_layout_generate_xml_before>
        </events>
    </global>
</config>

Observer.php

<!-- language: lang-php -->
<?php

class MyCompany_MyModule_Model_Observer
{
    public function changeRequestLayout($observer)
    {
        if ($observer->getAction()->getRequest()->isAjax()) {
            Mage::getDesign()->setArea('frontend')->setPackageName('myTheme')->setTheme('default');
        }
    }
}

local.xml

<!-- language: lang-xml -->
<?xml version="1.0" encoding="UTF-8"?>
<layout version="0.1.0.0">
    <default>
        <block type="page/html" name="root" output="toHtml" template="test.phtml" />
    </default>
</layout>

test.phtml

<!-- language: lang-html -->
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Strict//EN"
        "http://www.w3.org/TR/xhtml1/DTD/xhtml1-strict.dtd">
<html xmlns="http://www.w3.org/1999/xhtml">
<head>
    <title>Testing, testing...1...2...3...</title>
    <style type="text/css">
        body {
            background-color:#f00;
        }
    </style>
</head>
<body>
    <h1>This is a test of the Magento layout system.  This is only a test.  If this were not a test, real content would follow.</h1>
</body>
</html>

MyCompany_MyModule.xml

<!-- language: lang-xml -->
<?xml version="1.0" encoding="UTF-8"?>
<config>
    <modules>
        <MyCompany_MyModule>
            <active>true</active>
            <codePool>local</codePool>
        </MyCompany_MyModule>
    </modules>
</config>

【问题讨论】:

  • 我会从“你的观察者是否被调用”开始。不知道有太多不同的方向来回答这个问题。
  • 是的。它正在被调用。

标签: magento layout magento-1.7


【解决方案1】:

这是我构建的一个工作解决方案,它肯定可以通过controller_front_init_before 事件切换设计包/主题。

当然你可以检查changeConfigNode()方法中的一些请求参数 Mage::app()-&gt;getRequest()-&gt;getParam('switch') 并附加请求参数 switch/1 以触发更改。

   <global>
    <events>
        <controller_front_init_before>
            <observers>
                <cartware_change_the_node>
                     <class>cartware_guest2customer/observer</class>
                     <method>changeConfigNode</method>
                </cartware_change_the_node>
            </observers>
        </controller_front_init_before>               
    </events>
   </global>

以及观察者方法:

public function changeConfigNode(){

    /** left for testing
    Mage::log( Mage::app()->getConfig()->getXmlString());
            **/

    Mage::app()->getConfig()->setNode('stores/default/design/package/name', 'enterprise');
    Mage::app()->getConfig()->setNode('stores/default/design/package/theme', 'enterprise');
    Mage::app()->getConfig()->setNode('stores/default/design/package/default_theme', 'enterprise');
    Mage::app()->getConfig()->setNode('stores/default/design/theme/default', 'enterprise');
    Mage::app()->getConfig()->setNode('stores/default/design/theme/layout', 'enterprise');
    Mage::app()->getConfig()->setNode('stores/default/design/theme/template', 'enterprise');



}

它没有额外的检查,但它可能会帮助你!祝你好运!

【讨论】:

  • 我向 Mage_Core_Block_Template::fetchView 添加了一个 echo 语句,以显示它正在尝试加载的视图文件。我尝试了您的建议和我的原始逻辑,并确定问题不在逻辑中,而是在我的模板中(我尝试了我的原始方法并成功切换到我之前安装的另一个模板)。出于某种原因,我不明白,当我尝试切换到我的最小模板时,它会尝试加载不属于我的主题的 frontend/base/default/template/page/2columns-right.phtml。
  • 也许您的主题中没有 2columns-right 模板,因此回退到基本/默认。
  • 你是对的,我没有 2columns-right 模板。原因是我不希望我的网站使用它或任何其他内置模板。我希望 Magento 使用 my 模板,它不以任何形状或形式引用 2columns-right (至少,我看不到)。这就是为什么我正在寻找一种方法来覆盖 整个 布局。
  • 2-columns-right 用作“根”模板。估计还在用!您需要通过 layout.xml 和 setTemplate 阻止 reference the root 到另一个 2-columns-right.phtml
  • 那么...这与 local.xml 所做的有什么不同?
【解决方案2】:

真正的问题

Magento 的分层 XML 布局系统是在完成本应是微不足道的用例时造成混乱的根源。

因为基本/默认布局 XML 与我指定的任何布局相结合,所以我不仅必须覆盖根布局,还必须覆盖 base/default/page.xml 中定义的所有其他派生布局。这是不切实际的,因为 Magento 会根据客户端正在浏览的页面类型定义大量自定义布局。

我的解决方案

最简洁的解决方案是在模板/页面目录中创建符号链接,以指向我的 test.phtml 文件以用于以下所有布局。这保证了 Magento 将只使用 my 布局模板。

注意:您必须启用模板符号链接才能工作(在管理后端,导航到系统>配置>高级>开发人员,展开“模板设置”并将“允许符号链接”设置为“是”)。

链接到 test.phtml 的文件

  • 1column.phtml
  • 2columns-left.phtml
  • 2columns-right.phtml
  • 3columns.phtml
  • empty.phtml
  • popup.phtml
  • print.phtml
  • redirect.phtml

【讨论】:

  • 您的解决方案不好。当然,您可以通过更改根模板返回任何您想要的页面并完全跳过基本布局 - 如果您做得正确的话。这在处理 ajax 输出时最常使用。另一种解决方案是仅获取一个块并通过控制器操作中的响应返回其 html。您的 local.xml 完全错误,您应该使用 action method="setTemplate" 更改根模板。就像在您的根模板中一样,内容页眉页脚等没有 childhtml,它将永远不会被渲染...长话短说:您的解决方案不是正确的解决方案
  • @MischaLeiss 简要地说:是的,这是针对 AJAX 的。不,引用根目录并仅更改模板是 NOT 我需要的。整个问题的重点是找出如何正确替换/绕过整个布局。如果您有如何正确更改根模板的特定代码示例,我很乐意看到它。
猜你喜欢
  • 2016-04-21
  • 2015-03-13
  • 2018-07-17
  • 1970-01-01
  • 2011-02-25
  • 1970-01-01
  • 2011-03-10
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多