1

まず第一に、私はフランス人なので、私の英語がそれほど悪くないことを願っています^^' Symfony3 に問題があり、解決方法がわかりません。

3 つのドロップダウン リストを含むフォームがあります。2 つのテーマが関連しています: エージェンシーとセンター。1 つの機関が複数のセンターを持つことができます。そのため、代理店を選択すると、センターのリストが更新されます。3 番目のドロップダウン リストは [カテゴリ] です。それらはすべて同じ名前のエンティティに関連しています。

Symfony でこれを行うには、公式ドキュメントに従いました: http://symfony.com/doc/3.0/form/dynamic_form_modification.html#form-events-submitted-data

作成フォームではすべてが正常に機能します。しかし、編集に同じフォームを使用し、同じドロップダウン リストを使用すると、ajax リクエストが機能しません:/ カテゴリ (3 番目のドロップダウン リスト) が null に設定されていると表示されます。

キャッチされない PHP 例外 Symfony\Component\PropertyAccess\Exception\InvalidArgumentException: "Expected argument of type "EDVS\SinistreBundle\Entity\Category", "NULL" given" at C:\wamp\www\SinistraV2\vendor\symfony\symfony\src \Symfony\Component\PropertyAccess\PropertyAccessor.php 行 254

他の2つのドロップダウンリストとは関係ありませんが、どうやらこれが問題です。明確にするために、編集フォームで代理店を選択すると、このエラーがトリガーされます。これを行うと Ajax リクエストが開始され、リストがセンターで更新されるはずですが、そうではありません。

スタック トレースを見ると、setCategory() 関数で渡されたパラメーターが null です。

コントローラーと formType ファイルで確認しましたが、カテゴリが null ではありません。ドロップダウン リストには利用可能なカテゴリが表示され、データベースに保存されているカテゴリに設定されます。

ここで、私の「formType」ファイル:

    class SinistreType extends AbstractType
    {
        /**
         * @param FormBuilderInterface $builder
         * @param array $options
         */
        public function buildForm(FormBuilderInterface $builder, array $options)
        {
            // Récupération des sous-types, passés en paramètre via les options du form (voir aussi function configureOptions plus bas)
            $sousTypeSinistre = $options['sousTypeSinistre'];
            $em = $options['em'];

            $builder
            [...]
                ->add('categorie', EntityType::class, array(
                    'class'         => 'EDVSSinistreBundle:Categorie',
                    'placeholder'   => 'Choisir une catégorie',
                    'choice_label'  => 'intituleCat',
                ))
                [...]
                ->add('save', SubmitType::class);

            ;

            /**
             * Liste déroulante dynamique
             * Selon l'agence sélectionnée, la liste de centres correspondant change
             */
            $centresModifier = function (FormInterface $form, Agence $agence = null) {
                /**
                 * Si l'agence passée en param est null, $centres reçoit un tableau vide,
                 * sinon $centres prend comme valeur la liste des centres rattachés à l'agence
                 */
                $centres = null === $agence ? array() : $agence->getCentres();

                // Ajout, dans le formulaire, de la liste déroulante contenant les centres récupérés précédemment
                $form->add('centre', EntityType::class, array(
                    'class'         => 'EDVSAgenceCentreBundle:Centre',
                    'placeholder'   => 'Choisir un centre',
                    'choices'       => $centres,
                    'choice_label'  => 'nom',
                ));
            };

            // Evénement appelé au moment de la construction du formulaire
            $builder->addEventListener(FormEvents::PRE_SET_DATA, function (FormEvent $event) use ($centresModifier, $em) {
                $data = $event->getData(); // Entité Sinistre
                $form = $event->getForm();

                // Dans le cas d'une modification (données provenant de la BDD)
                if ($data->getCentreUtilise()) {
                    // Récupération du centre & de l'agence par rapport au centreUtilisé & à l'agenceUtilisée liés au dossier en cours de modification
                    $centre = $em->getRepository('EDVSAgenceCentreBundle:Centre')->getCentreByNom($data->getCentreUtilise()->getNom());
                    $agence = $em->getRepository('EDVSAgenceCentreBundle:Agence')->getAgenceByNom($data->getCentreUtilise()->getAgenceUtilisee()->getNom());

                    // Sélection de l'agence & du centre dans les listes déroulantes correspondantes
                    $data->setCentre($centre[0]);
                    $data->setAgence($agence[0]);

                    // Affichage de la liste des agences disponibles & des centres correspondants
                    $centresModifier($form, $data->getAgence());
                } else { // Dans le cas d'une création (données vides)
                    $centresModifier($form, $data->getAgence());
                }
            });

            // Evénement appelé juste après que le formulaire ait été validé, concerne le champ "agence" seulement
            $builder->get('agence')->addEventListener(FormEvents::POST_SUBMIT, function(FormEvent $event) use ($centresModifier) {
            $form = $event->getForm();
            $agence = $form->getData();

            $centresModifier($form->getParent(), $agence);
        });
}

/**
 * @param OptionsResolver $resolver
 */
public function configureOptions(OptionsResolver $resolver)
{
    $resolver->setDefaults(array(
        'data_class'        => 'EDVS\SinistreBundle\Entity\Sinistre',
        'sousTypeSinistre'  => null, // Déclaration du paramètre "typeSinistre" pour qu'il puisse être reconnu comme option (transfert du param controller vers form)
        'em'                => null, // Idem
    ));
}

}

JavaScript部分:

$(document).ready(function() {
    var agence = $('#sinistre_edit_agence');

    /* Actualisation de la liste déroulante des centres en fonction de l'agence sélectionnée */
    agence.change(function() {
        // Animation "Chargement en cours"
        toggleLoading();

        var form = $(this).closest('form');
        var data = {};
        data[agence.attr('name')] = agence.val();

        $.ajax({
            url : form.attr('action'),
            type: form.attr('method'),
            data : data,
            success: function(html) {
                $('#sinistre_edit_centre').replaceWith(
                    $(html).find('#sinistre_edit_centre')
                );
                toggleLoading();
            },
            error: function(error) {
                console.error(error);
                toggleLoading();
            }
        });
    });
});

ここでも同じ問題のようです: Symfony 2: 動的フォームイベントは、編集時にのみ InvalidArgumentException を返します

しかし、答えはなく、何が間違っているのか本当にわかりません。フォームイベントを正しく使用していないと思います。これは私にとってまったく新しいことです。

私はここでいくつかの助けを見つけることを願っています, 前もって感謝!

4

1 に答える 1

0

わかりましたので、解決策を見つけました。どうやら問題は私のjavascriptにあったようです。Ajax リクエストの属性「data」には、ドロップダウン リスト「Category」の値を渡す必要があります。次に、Ajax リクエストを実行すると、正常に動作します。そうしないと、最初のメッセージで述べたように、"Category" 値が null に設定されているため、Ajax リクエストは失敗します。

私の新しいJavaScript:

$(document).ready(function() {
var agence = $('#sinistre_edit_agence');

/* Actualisation de la liste déroulante des centres en fonction de l'agence sélectionnée */
agence.change(function() {
    // Animation "Chargement en cours"
    toggleLoading();

    var form = $(this).closest('form');
    var data = {};
    data[agence.attr('name')] = agence.val();
    data[categorie.attr('name')] = categorie.val(); // FIX : the category value

    $.ajax({
        url : form.attr('action'),
        type: form.attr('method'),
        data : data, // Now, contains my "Agence" value and "Category" value
        success: function(html) {
            $('#sinistre_edit_centre').replaceWith(
                $(html).find('#sinistre_edit_centre')
            );
            toggleLoading();
        },
        error: function(error) {
            console.error(error);
            toggleLoading();
        }
    });
});

});

他の 2 つの関連ドロップダウン リストとは関係のない 3 番目のドロップダウン リストが問題であった理由がよくわからないことを認めなければなりません。

于 2016-09-14T09:19:29.123 に答える