Doctrine2-Spatialを扱うときの注意点

Doctrine2-Spatialを扱うときの注意点

July 17, 2018,
tags: php doctrine mysql symfony


このエントリーをはてなブックマークに追加

Doctrineで緯度経度を扱いたいときに便利なライブラリ「Doctrine2-Spatial」 GitHub - creof/doctrine2-spatial: Doctrine2 multi-platform support for spatial types and functions.

基本的な使い方は以下のサイトが非常に参考になる。
SymfonyとDoctrine ORMでMySQLの空間インデックス(SPATIAL INDEX)を扱う方法 | ホームページ作成、システム設計 glic株式会社

今回僕が遭遇した問題は、「更新時に緯度経度の情報を変更しても、保存されない」という問題に遭遇した。
FormTypeはこんな感じ。

<?php

namespaceApp\Form\Type;


useCrEOF\Spatial\PHP\Types\Geometry\Point;
useSymfony\Component\Form\AbstractType;
useSymfony\Component\Form\Extension\Core\Type\TextType;
useSymfony\Component\Form\FormBuilderInterface;
useSymfony\Component\OptionsResolver\OptionsResolver;

class PointType extendsAbstractType
{
    public functionbuildForm(FormBuilderInterface $builder, array $options)
    {
        $builder->add('x', TextType::class,[
            'label' => '緯度',
            'required' => true,
        ]);

        $builder->add('y', TextType::class,[
            'label' => '経度',
            'required' => true,
        ]);
    }


    public functionconfigureOptions(OptionsResolver $resolver)
    {
        $resolver->setDefaults([
            'data_class' => Point::class,
            'translation_domain' => false,
        ]);
    }

}

Pointオブジェクトが同じオブジェクトだとなぜか緯度経度の情報が変更されても、変更されない。
DataTransformerでオブジェクトをcloneしてあげたら無事に保存された。
ちゃんと実装コードを追っていないので、原因は正確に把握してないけど、とりあえずcloneすればいいみたい。

<?php

namespaceApp\Form\Type;


useCrEOF\Spatial\PHP\Types\Geometry\Point;
useSymfony\Component\Form\AbstractType;
useSymfony\Component\Form\CallbackTransformer;
useSymfony\Component\Form\Extension\Core\Type\TextType;
useSymfony\Component\Form\FormBuilderInterface;
useSymfony\Component\OptionsResolver\OptionsResolver;

class PointType extendsAbstractType
{
    public functionbuildForm(FormBuilderInterface $builder, array $options)
    {
        $builder->add('x', TextType::class,[
            'label' => '緯度',
            'required' => true,
        ]);

        $builder->add('y', TextType::class,[
            'label' => '経度',
            'required' => true,
        ]);


        $builder->addModelTransformer(newCallbackTransformer(function($data){
            return$data;
        }, function($data){
            return clone $data;
        }));
    }


    public functionconfigureOptions(OptionsResolver $resolver)
    {
        $resolver->setDefaults([
            'data_class' => Point::class,
            'translation_domain' => false,
        ]);
    }

}

これで、保存される。
たまに引っかかるので気をつけてください。

comments powered by Disqus