5

Sto imparando drupal 8. Voglio creare una pagina che contenga un modulo 'two dimensionalnal' 'aggiungi un altro oggetto'. Il mio codice funziona bene, ma ho uno strano comportamento quando aggiungo stanze a una casa (c'è un valore strano nei miei registri di debug da FormStateInterface :: getTriggeringElement(), vedere in fondo per il codice e il registro)Problema per "due dimensioni" aggiungi un altro elemento "" con FormStateInterface :: getTriggeringElement()

Primo: ho due strutture, case e stanze. L'utente può creare alcune case e per ogni casa, si può creare alcune stanze:

enter image description here

Quando aggiungo alcune case, la forma funziona bene:

enter image description here

Quando aggiungo un po ' sale per l'ultima casa, il modulo funziona bene troppo:

enter image description here

Ma quando aggiungo alcune stanze a una casa "senza fine", il modulo non funziona correttamente (nello screenshot, faccio clic una volta su "aggiungi stanza" nel blocco "1", l'etichetta del "casa 1" è diventato "casa 2" e fare clic su aggiungi 5 camere (?!) (?!):

enter image description here

Ecco il mio codice e una strana registro di debug, che non spiegano perché ho questo valore (dalla getTriggeringElement() nella richiamata room_addMoreSubmit e questo è il problema credo)

<?php 

namespace Drupal\projet\Form; 
use Drupal\Core\Form\FormBase; 
use Drupal\Core\Form\FormStateInterface; 

class HouseForm extends FormBase { 

    public function getFormId(){ 
    return 'custom_rooms_form'; 
    } 

    function buildForm(array $form, FormStateInterface $form_state) { 



    $house_count = $form_state->get('house_count'); 

    if (is_null($house_count)) { 
     $house_count = 1; 
     $form_state->set('house_count', $house_count); 
    } 

    $form['house'] = array(
     //'#tree' => TRUE, 
     '#prefix' => '<div id="house-replace">', 
     '#suffix' => '</div>' 
    ); 

    for ($house_delta = 0; $house_delta < $house_count; $house_delta++) { 
     if (!isset($form['house'][$house_delta])) { 

     $room_count[$house_delta] = $form_state->get('room_count_'.$house_delta); 

     if (is_null($room_count[$house_delta])) { 
      $room_count[$house_delta] = 1; 
      $form_state->set('room_count_'.$house_delta, $room_count[$house_delta]); 
     } 

     dd($room_count, "room_COUNT"); 

     $form['house'][$house_delta]['room'] = array(
      '#type' => 'fieldset', 
      '#title' => t('house : '.$house_delta), 
      //'#tree' => TRUE, 
      '#prefix' => '<div id="room-replace-'.$house_delta.'">', 
      '#suffix' => '</div>' 
     ); 

     for ($room_delta = 0; $room_delta < $room_count[$house_delta]; $room_delta++) { 
      if (!isset($form['house'][$house_delta]['room'][$room_delta])) { 
      $room = array(
       '#type' => 'textfield' 
      ); 
      $check = array(
       '#type' => 'checkbox' 
      ); 
      $form['house'][$house_delta]['room'][$room_delta] = array(
       '#type' => 'fieldset', 
       '#title' => t('room : '.$house_delta.'.'.$room_delta), 
      ); 
      $form['house'][$house_delta]['room'][$room_delta]['text'] = $room; 
      $form['house'][$house_delta]['room'][$room_delta]['check'] = $check; 
      } 
     } 

     $form['house'][$house_delta]['room']['add'] = array(
      '#type' => 'submit', 
      '#name' => 'add', 
      '#value' => t('Add room'), 
      '#attributes' => array('class' => array('field-add-more-submit'), 'house_delta' => array($house_delta)), 
      '#submit' => array(array(get_class($this), 'room_addMoreSubmit')), 
      '#ajax' => array(
       'callback' => array($this, 'room_addMoreCallback'), 
       'wrapper' => 'room-replace-'.$house_delta, 
       'effect' => 'fade', 
      ), 
     ); 

     } 
    } 

    $form['house']['add'] = array(
     '#type' => 'submit', 
     '#name' => 'add', 
     '#value' => t('Add house'), 
     '#attributes' => array('class' => array('field-add-more-submit')), 
     '#submit' => array(array(get_class($this), 'house_addMoreSubmit')), 
     '#ajax' => array(
      'callback' => array($this, 'house_addMoreCallback'), 
      'wrapper' => 'house-replace', 
      'effect' => 'fade', 
     ), 
    ); 


    $form['submit'] = array(
     '#type' => 'submit', 
     '#value' => t('Create'), 
    ); 

    return $form; 
    } 

    public function room_addMoreSubmit(array $form, FormStateInterface $form_state) { 
    dd($form_state->getTriggeringElement(), "room : getTriggeringElement()"); // below, the log when I add a room to the house '1' (result see above with the last screenshot: "the house 1" became "house 2" and one click add 5 rooms) 
    $house = $form_state->getTriggeringElement()["#array_parents"][1]; 
    $c = $form_state->get('room_count_'.$house) + 1; 
    $form_state->set('room_count_'.$house, $c); 
    $form_state->setRebuild(TRUE); 
    } 

    public function room_addMoreCallback(array $form, FormStateInterface $form_state) { 
    $house = $form_state->getTriggeringElement()["#array_parents"][1]; 
    return $form['house'][$house]['room']; 
    } 

    public function house_addMoreSubmit(array $form, FormStateInterface $form_state) { 
    dd($form_state->getTriggeringElement()["#array_parents"], "house : getTriggeringElement()"); 
    $c = $form_state->get('house_count') + 1; 
    $form_state->set('house_count', $c); 
    $form_state->setRebuild(TRUE); 
    } 

    public function house_addMoreCallback(array $form, FormStateInterface $form_state) { 
    return $form['house']; 
    } 

} 

il registro ('dd' nel room_addMoreSubmit) quando si fa clic sul pulsante "aggiungi stanza" in la casa "1":

enter image description here

Quando clicco sul pulsante "aggiungi stanza" nel numero civico 1, getTriggeringElement restituire i genitori matrice del pulsante Aggiungi. E, come puoi vedere, il genitore è "2" non "1" (la casa 1) Quindi quando clicco sul pulsante "aggiungi stanza" della casa 1, questa è la casa "2" che è identificata e non la casa "1".

Non capisco perché ... Utilizzare il getTriggeringElement non è il modo migliore?

risposta

3

La soluzione:

$form['house'][$house_delta]['room']['add'] = array(
      '#type' => 'submit', 
      '#name' => 'add-'.$house_delta, 
      '#value' => t('Add room'), 
      '#attributes' => array('class' => array('field-add-more-submit'), 'house_delta' => array($house_delta)), 
      '#submit' => array(array(get_class($this), 'room_addMoreSubmit')), 
      '#ajax' => array(
       'callback' => array($this, 'room_addMoreCallback'), 
       'wrapper' => 'room-replace-'.$house_delta, 
       'effect' => 'fade', 
      ), 
     ); 

Il nome univoco è stata la questione del mio problema. Quindi, cambio l'attributo name.