'submit', // Hide this button when JavaScript is enabled. '#attributes' => array('class' => array('js-hide')), '#submit' => array('views_ui_nojs_submit'), // Add a process function to limit this button's validation errors to the // triggering element only. We have to do this in #process since until the // form API has added the #parents property to the triggering element for // us, we don't have any (easy) way to find out where its submitted values // will eventually appear in $form_state['values']. '#process' => array_merge(array('views_ui_add_limited_validation'), element_info_property('submit', '#process', array())), // Add an after-build function that inserts a wrapper around the region of // the form that needs to be refreshed by AJAX (so that the AJAX system can // detect and dynamically update it). This is done in #after_build because // it's a convenient place where we have automatic access to the complete // form array, but also to minimize the chance that the HTML we add will // get clobbered by code that runs after we have added it. '#after_build' => array_merge(element_info_property('submit', '#after_build', array()), array('views_ui_add_ajax_wrapper')), ); // Copy #weight and #access from the triggering element to the button, so // that the two elements will be displayed together. foreach (array('#weight', '#access') as $property) { if (isset($triggering_element[$property])) { $wrapping_element[$button_key][$property] = $triggering_element[$property]; } } // For easiest integration with the form API and the testing framework, we // always give the button a unique #value, rather than playing around with // #name. $button_title = !empty($triggering_element['#title']) ? $triggering_element['#title'] : $trigger_key; if (empty($seen_buttons[$button_title])) { $wrapping_element[$button_key]['#value'] = t('Update "@title" choice', array( '@title' => $button_title, )); $seen_buttons[$button_title] = 1; } else { $wrapping_element[$button_key]['#value'] = t('Update "@title" choice (@number)', array( '@title' => $button_title, '@number' => ++$seen_buttons[$button_title], )); } // Attach custom data to the triggering element and submit button, so we can // use it in both the process function and AJAX callback. $ajax_data = array( 'wrapper' => $triggering_element['#ajax']['wrapper'], 'trigger_key' => $trigger_key, 'refresh_parents' => $refresh_parents, // Keep track of duplicate wrappers so we don't add the same wrapper to the // page more than once. 'duplicate_wrapper' => !empty($seen_ids[$triggering_element['#ajax']['wrapper']]), ); $seen_ids[$triggering_element['#ajax']['wrapper']] = TRUE; $triggering_element['#views_ui_ajax_data'] = $ajax_data; $wrapping_element[$button_key]['#views_ui_ajax_data'] = $ajax_data; } /** * Processes a non-JavaScript fallback submit button to limit its validation errors. */ function views_ui_add_limited_validation($element, &$form_state) { // Retrieve the AJAX triggering element so we can determine its parents. (We // know it's at the same level of the complete form array as the submit // button, so all we have to do to find it is swap out the submit button's // last array parent.) $array_parents = $element['#array_parents']; array_pop($array_parents); $array_parents[] = $element['#views_ui_ajax_data']['trigger_key']; $ajax_triggering_element = NestedArray::getValue($form_state['complete_form'], $array_parents); // Limit this button's validation to the AJAX triggering element, so it can // update the form for that change without requiring that the rest of the // form be filled out properly yet. $element['#limit_validation_errors'] = array($ajax_triggering_element['#parents']); // If we are in the process of a form submission and this is the button that // was clicked, the form API workflow in form_builder() will have already // copied it to $form_state['triggering_element'] before our #process // function is run. So we need to make the same modifications in $form_state // as we did to the element itself, to ensure that #limit_validation_errors // will actually be set in the correct place. if (!empty($form_state['triggering_element'])) { $clicked_button = &$form_state['triggering_element']; if ($clicked_button['#name'] == $element['#name'] && $clicked_button['#value'] == $element['#value']) { $clicked_button['#limit_validation_errors'] = $element['#limit_validation_errors']; } } return $element; } /** * After-build function that adds a wrapper to a form region (for AJAX refreshes). * * This function inserts a wrapper around the region of the form that needs to * be refreshed by AJAX, based on information stored in the corresponding * submit button form element. */ function views_ui_add_ajax_wrapper($element, &$form_state) { // Don't add the wrapper
if the same one was already inserted on this // form. if (empty($element['#views_ui_ajax_data']['duplicate_wrapper'])) { // Find the region of the complete form that needs to be refreshed by AJAX. // This was earlier stored in a property on the element. $complete_form = &$form_state['complete_form']; $refresh_parents = $element['#views_ui_ajax_data']['refresh_parents']; $refresh_element = NestedArray::getValue($complete_form, $refresh_parents); // The HTML ID that AJAX expects was also stored in a property on the // element, so use that information to insert the wrapper
here. $id = $element['#views_ui_ajax_data']['wrapper']; $refresh_element += array( '#prefix' => '', '#suffix' => '', ); $refresh_element['#prefix'] = '
' . $refresh_element['#prefix']; $refresh_element['#suffix'] .= '
'; // Copy the element that needs to be refreshed back into the form, with our // modifications to it. NestedArray::setValue($complete_form, $refresh_parents, $refresh_element); } return $element; } /** * Updates a part of the add view form via AJAX. * * @return * The part of the form that has changed. */ function views_ui_ajax_update_form($form, $form_state) { // The region that needs to be updated was stored in a property of the // triggering element by views_ui_add_ajax_trigger(), so all we have to do is // retrieve that here. return NestedArray::getValue($form, $form_state['triggering_element']['#views_ui_ajax_data']['refresh_parents']); } /** * Non-Javascript fallback for updating the add view form. */ function views_ui_nojs_submit($form, &$form_state) { $form_state['rebuild'] = TRUE; } /** * Form element validation handler for a taxonomy autocomplete field. * * This allows a taxonomy autocomplete field to be validated outside the * standard Field API workflow, without passing in a complete field widget. * Instead, all that is required is that $element['#field_name'] contain the * name of the taxonomy autocomplete field that is being validated. * * This function is currently not used for validation directly, although it * could be. Instead, it is only used to store the term IDs and vocabulary name * in the element value, based on the tags that the user typed in. * * @see taxonomy_autocomplete_validate() */ function views_ui_taxonomy_autocomplete_validate($element, &$form_state) { $value = array(); if ($tags = $element['#value']) { // Get the machine names of the vocabularies we will search, keyed by the // vocabulary IDs. $field = field_info_field($element['#entity_type'], $element['#field_name']); $vocabularies = array(); $allowed_values = $field->getSetting('allowed_values'); if (!empty($allowed_values)) { foreach ($allowed_values as $tree) { if ($vocabulary = entity_load('taxonomy_vocabulary', $tree['vocabulary'])) { $vocabularies[$vocabulary->id()] = $tree['vocabulary']; } } } // Store the term ID of each (valid) tag that the user typed. $typed_terms = drupal_explode_tags($tags); foreach ($typed_terms as $typed_term) { if ($terms = entity_load_multiple_by_properties('taxonomy_term', array('name' => trim($typed_term), 'vid' => array_keys($vocabularies)))) { $term = array_pop($terms); $value['tids'][] = $term->id(); } } // Store the term IDs along with the name of the vocabulary. Currently // Views (as well as the Field UI) assumes that there will only be one // vocabulary, although technically the API allows there to be more than // one. if (!empty($value['tids'])) { $value['tids'] = array_unique($value['tids']); $value['vocabulary'] = array_pop($vocabularies); } } form_set_value($element, $value, $form_state); } /** * Add a