我正在尝试使用magento中的观察者根据请求中的参数完全替换给定请求的整个布局。
我面临的问题是Magento仍在尝试加载与我在Observer中指定的模板不同的根块模板(特别是产品页面的“frontend / base / default / template / page / 1column.phtml”)这是在后端的设计选项卡中配置的模板使用的默认根块模板。因为它没有使用我指定的布局,所以它在PHP中死亡,试图加载主题中不存在的模板。
感谢任何指导。
注意:我没有使用CMS页面来测试这个概念,因为它们加载了后端指定的模板。我创建了一个测试产品并使用了它的产品页面。通过请求以下网址进行测试:http://mymagentosite.com/test-product?ajax=1
我创建了一个带有观察者和非常小布局的模块,如下所示:
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
<!-- 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>
<!-- 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');
}
}
}
<!-- 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>
<!-- 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>
<!-- language: lang-xml -->
<?xml version="1.0" encoding="UTF-8"?>
<config>
<modules>
<MyCompany_MyModule>
<active>true</active>
<codePool>local</codePool>
</MyCompany_MyModule>
</modules>
</config>
答案 0 :(得分:1)
这是我构建的工作解决方案,它确实可以通过controller_front_init_before
事件切换设计包/主题。
当然你可以用changeConfigNode()
方法检查一些请求参数
Mage::app()->getRequest()->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');
}
它没有额外的检查,但它可能会帮助你!祝你好运!
答案 1 :(得分:0)
Magento的分层XML布局系统是实现应该是一个微不足道的用例的混乱的根源。
因为基本/默认布局XML与我指定的任何布局相结合,所以我不仅要覆盖根布局,还要覆盖base / default / page.xml中定义的所有其他派生布局。这是不切实际的,因为Magento根据客户端浏览的页面类型定义了大量自定义布局。
最干净的解决方案是在模板/页面目录中创建符号链接,指向我的test.phtml文件,以获取以下所有布局。这可以保证Magento只使用我的布局模板。
注意:您必须启用模板符号链接才能工作(在管理员后端,导航到系统&gt;配置&gt;高级&gt;开发人员,展开“模板设置”并将“允许符号链接”设置为“是”)。