Doctrine 1.2自动加入i18n?

时间:2011-03-25 17:14:20

标签: php oop orm internationalization doctrine


我想扩展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并获取带有加载翻译的产品?

1 个答案:

答案 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对象所需的数据操作类(控制器)面向应用程序架构...