没有任何东西将数据传输到块。在控制器动作完成其模型交互后,它负责
加载布局对象(间接加载和创建块对象)
告诉该布局对象呈现一个页面。
大多数 Magento 控制器操作在控制器操作结束时通过两次调用来完成此操作。
$this->loadLayout();
$this->renderLayout();
在 Magento 中,视图上没有设置数据。相反,视图(即块对象)向系统询问数据。您可以在 Mage_Tag_Block_Customer_View 块类中看到这样的示例。
#File: app/code/core/Mage/Tag/Block/Customer/View.php
...
public function getTagInfo()
{
if (is_null($this->_tagInfo)) {
$this->_tagInfo = Mage::getModel('tag/tag')
->load($this->getTagId());
}
return $this->_tagInfo;
}
...
这里,这个块的getTagInfo 方法直接向模型询问它的信息。这样,前端模板开发者就可以访问一个
$this->getTagInfo();
方法。我还在good authority 上指出,块的_prepareLayout 方法是将大部分(如果不是全部)数据获取代码放入块中的理想场所。
您将看到使用的第二种模式是 Magento 注册表模式。这是一个 Magento 系统,可让您设置系统范围(但不是 PHP)的全局变量。
Mage::register('foo', 'some value');
echo Mage::registry('foo');
有时,Magento 开发人员会使用注册表在控制器操作中设置变量,然后在块中重新抓取。例如,在管理控制台的发票控制器中。
#File: app/code/core/Mage/Adminhtml/controllers/Sales/Order/InvoiceController.php
protected function _initInvoice()
{
...
$invoice = Mage::register('current_invoice', $invoice);
return $invoice;
}
然后一个 Block 稍后会引用它。
#File: app/code/core/Mage/Sales/Block/Order/Print/Invoice.php
public function getInvoice()
{
return Mage::registry('current_invoice');
}
我对注册表模式并不感兴趣,但它被核心团队使用,所以它可能是犹太教的。
最后,如果您希望模拟大多数 PHP MVC 框架中使用的“哑视图”模式,请尝试类似的方法
$this->loadLayout();
$block = $this->getLayout()->getBlock('block_name');
$block->setSomeData('My Data');
$block->setData('alternate_syntax', 'Some other data');
$this->renderLayout();
然后在块和/或模板文件中。
echo $this->getSomeData();
echo $this->getData('some_data');
echo $this->getAlternateSyntax();
echo $this->getData('alternate_syntax');
调用loadLayout 后,Magento 将创建所有块对象。您在上面所做的是获取对特定块对象的引用,然后设置其数据。
根据下面 Vinai 的 cmets,还有一个块的 assign 方法需要考虑。
类似于setData,在调用loadLayout(或从一个块的_prepareLayout)方法之后,你可以做类似的事情
$this->loadLayout();
$block = $this->getLayout()->getBlock('block_name');
$block->assign('my_view_var','Something for the view');
$this->renderLayout();
然后在块的phtml 文件中,您就可以输出该视图变量
echo $my_view_var;