我想扩展i18n行为,以便它在任何类型的查询(DQL,relations,getTable)上自动加入转换表。
此外,它需要定义默认语言参数,因此当我在没有语言设置的情况下进行查询时,它会回退到默认语言。
注意:我正在寻找一个通用行为,因此这适用于所有i18n模型对象,而不是为每个类编写和覆盖。
这是一个例子:
表product
- > id,category_id,price ...
表product_translation
- > id,lang,name,description ...
使用当前的解决方案,当我这样做:Doctrine_Core::getTable('Product')->findAll()
时,它会在不加入翻译的情况下获得所有产品。
所以在控制器中我必须循环遍历所有记录并重新应用翻译的值,$product->name = $product->Translation['en']->name
我想要这样的事情:
Doctrine_Core::getTable('Product')->findAll()
它应该获取lang ='en'Doctrine_Core::getTable('Product)->findAll('en')
与上述相同$user->Products
的类User,它应该返回包含翻译的集合。$user->Products('en')
之类的内容也应该返回其他(非默认)语言的集合Doctrine_Core::getTable('Product')->getByCategoryAndLang(1,'en')
有人可以帮忙吗?我正在看模板和行为,我认为这是要走的路,但不知道如何实现这个
编辑:我看到对此没什么兴趣,所以让我尝试一个更简单的问题。你通常如何通过关系获得i18n领域。例如,我如何调用$user->Products
并获取带有加载翻译的产品?
答案 0 :(得分:1)
我认为除非你想要全自动,否则你不需要扩展标准的Doctrine行为。但是你仍然可以像我们一样尝试这样做 - 我们使用DAO(数据访问对象)返回我们具体的Doctrine实体(Doctrine表表示):
\DAO::get('Some\Namespace\Classname')
其中Classname
代表PHP类模型描述的表。我们的DAO类创建了Classname
的实例,该实例封装在proxy
中(参见设计模式)。
除了表类模型之外,我们为该表创建另一个类,它位于表模型之上并使用此模型进行操作。在这个课程中,我们编写了getProducts($args)
,getProduct($id)
,getProductsByCategory($catId)
等方法。
我认为这就是你要找的......
在方法getProducts($args)
中,您可以在DQL中实现->leftJoin()
,它将通过$lang
参数中给定的$args
标识符加入转换表。简单的例子(未经测试):
class Products extends \DAO {
public function save($item) {
$item->save();
}
public function getProducts($args = array()) {
$order = array('p.id');
$result = \Doctrine_Query::create()
->from('Some\Namespace\Product p')
->where('1 = 1');
if(!empty($args['lang'])) {
$result = $result->leftJoin('Some\Namespace\ProductTranslation pt ON pt.product_id = p.id AND pt.language = ?', $args['lang']);
}
$result = $result->orderBy($order);
$result = $result->execute();
return $result;
}
}
然后致电
$products = DAO::get('Some\Namespace\Product')->getProducts(array('lang' => 1));
您获得所有加载了英文翻译的产品......
它不是自动化的,你必须为每个模型编写自己的DAO类,但这是一个很好的方法,因为你有不同的数据定义类(模型)和MVC / MVP对象所需的数据操作类(控制器)面向应用程序架构...