$languages, '#language_default' => $default, '#type' => 'table', '#header' => array( t('Name'), t('Weight'), t('Operations'), ), '#tabledrag' => array( array('order', 'sibling', 'language-order-weight'), ), ); foreach ($languages as $langcode => $language) { $form['languages'][$langcode]['#attributes']['class'][] = 'draggable'; $form['languages'][$langcode]['name'] = array( '#markup' => check_plain($language->name), ); $form['languages'][$langcode]['weight'] = array( '#type' => 'weight', '#title' => t('Weight for @title', array('@title' => $language->name)), '#title_display' => 'invisible', '#default_value' => $language->weight, '#attributes' => array( 'class' => array('language-order-weight'), ), '#delta' => 30, ); $links = array(); $links['edit'] = array( 'title' => t('edit'), 'href' => 'admin/config/regional/language/edit/' . $langcode, ); if ($langcode != $default->id) { $links['delete'] = array( 'title' => t('delete'), 'href' => 'admin/config/regional/language/delete/' . $langcode, ); } $form['languages'][$langcode]['operations'] = array( '#type' => 'operations', '#links' => $links, '#weight' => 100, ); } $form['actions'] = array('#type' => 'actions'); $form['actions']['submit'] = array( '#type' => 'submit', '#value' => t('Save configuration'), ); return $form; } /** * Process language overview form submissions, updating existing languages. */ function language_admin_overview_form_submit($form, &$form_state) { $languages = language_list(); foreach ($languages as $langcode => $language) { $language->weight = $form_state['values']['languages'][$langcode]['weight']; language_save($language); } drupal_set_message(t('Configuration saved.')); } /** * User interface for the language addition screen. */ function language_admin_add_form($form, &$form_state) { $predefined_languages = language_admin_predefined_list(); $predefined_languages['custom'] = t('Custom language...'); $predefined_default = !empty($form_state['values']['predefined_langcode']) ? $form_state['values']['predefined_langcode'] : key($predefined_languages); $form['predefined_langcode'] = array( '#type' => 'select', '#title' => t('Language name'), '#default_value' => $predefined_default, '#options' => $predefined_languages, ); $form['predefined_submit'] = array( '#type' => 'submit', '#value' => t('Add language'), '#limit_validation_errors' => array(array('predefined_langcode'), array('predefined_submit')), '#states' => array( 'invisible' => array( 'select#edit-predefined-langcode' => array('value' => 'custom'), ), ), '#validate' => array('language_admin_add_predefined_form_validate'), '#submit' => array('language_admin_add_form_submit'), ); $form['custom_language'] = array( '#type' => 'container', '#states' => array( 'visible' => array( 'select#edit-predefined-langcode' => array('value' => 'custom'), ), ), ); _language_admin_common_controls($form['custom_language']); $form['custom_language']['submit'] = array( '#type' => 'submit', '#value' => t('Add custom language'), '#validate' => array('language_admin_add_custom_form_validate'), '#submit' => array('language_admin_add_form_submit'), ); return $form; } /** * Editing screen for a particular language. * * @param $langcode * Language code of the language to edit. */ function language_admin_edit_form($form, &$form_state, $language) { _language_admin_common_controls($form, $language); $form['actions'] = array('#type' => 'actions'); $form['actions']['submit'] = array( '#type' => 'submit', '#value' => t('Save language') ); $form['#submit'][] = 'language_admin_edit_form_submit'; $form['#validate'][] = 'language_admin_edit_form_validate'; return $form; } /** * Common elements of the language addition and editing form. * * @param $form * A parent form item (or empty array) to add items below. * @param $language * Language object to edit. */ function _language_admin_common_controls(&$form, $language = NULL) { if (!is_object($language)) { $language = new Language(array('id' => NULL, 'name' => NULL,)); } if (isset($language->id)) { $form['langcode_view'] = array( '#type' => 'item', '#title' => t('Language code'), '#markup' => $language->id ); $form['langcode'] = array( '#type' => 'value', '#value' => $language->id ); } else { $form['langcode'] = array( '#type' => 'textfield', '#title' => t('Language code'), '#maxlength' => 12, '#required' => TRUE, '#default_value' => '', '#disabled' => FALSE, '#description' => t('Use language codes as defined by the W3C for interoperability. Examples: "en", "en-gb" and "zh-hant".', array('@w3ctags' => 'http://www.w3.org/International/articles/language-tags/')), ); } $form['name'] = array( '#type' => 'textfield', '#title' => t('Language name'), '#maxlength' => 64, '#default_value' => @$language->name, '#required' => TRUE, ); $form['direction'] = array( '#type' => 'radios', '#title' => t('Direction'), '#required' => TRUE, '#description' => t('Direction that text in this language is presented.'), '#default_value' => @$language->direction, '#options' => array(Language::DIRECTION_LTR => t('Left to right'), Language::DIRECTION_RTL => t('Right to left')), ); return $form; } /** * Element specific validator for the Add language button. */ function language_admin_add_predefined_form_validate($form, &$form_state) { $langcode = $form_state['values']['predefined_langcode']; if ($langcode == 'custom') { form_set_error('predefined_langcode', t('Fill in the language details and save the language with Add custom language.')); } else { if ($language = language_load($langcode)) { form_set_error('predefined_langcode', t('The language %language (%langcode) already exists.', array('%language' => $language->name, '%langcode' => $langcode))); } } } /** * Validate the language addition form on custom language button. */ function language_admin_add_custom_form_validate($form, &$form_state) { if ($form_state['values']['predefined_langcode'] == 'custom') { $langcode = $form_state['values']['langcode']; // Reuse the editing form validation routine if we add a custom language. language_admin_edit_form_validate($form['custom_language'], $form_state); if ($language = language_load($langcode)) { form_set_error('langcode', t('The language %language (%langcode) already exists.', array('%language' => $language->name, '%langcode' => $langcode))); } } else { form_set_error('predefined_langcode', t('Use the Add language button to save a predefined language.')); } } /** * Process the custom and predefined language addition form submission. */ function language_admin_add_form_submit($form, &$form_state) { $langcode = $form_state['values']['predefined_langcode']; if ($langcode == 'custom') { $langcode = $form_state['values']['langcode']; // Custom language form. $language = new Language(array( 'id' => $langcode, 'name' => $form_state['values']['name'], 'direction' => $form_state['values']['direction'], )); } else { $language = new Language(array('id' => $langcode)); } // Save the language and inform the user that it happened. $language = language_save($language); drupal_set_message(t('The language %language has been created and can now be used.', array('%language' => $language->name))); // Tell the user they have the option to add a language switcher block // to their theme so they can switch between the languages. drupal_set_message(t('Use one of the language switcher blocks to allow site visitors to switch between languages. You can enable these blocks on the block administration page.', array('@block-admin' => url('admin/structure/block')))); $form_state['redirect'] = 'admin/config/regional/language'; } /** * Validate the language editing form. Reused for custom language addition too. */ function language_admin_edit_form_validate($form, &$form_state) { // Ensure sane field values for langcode and name. if (!isset($form['langcode_view']) && preg_match('@[^a-zA-Z_-]@', $form_state['values']['langcode'])) { form_set_error('langcode', t('%field may only contain characters a-z, underscores, or hyphens.', array('%field' => $form['langcode']['#title']))); } if ($form_state['values']['name'] != check_plain($form_state['values']['name'])) { form_set_error('name', t('%field cannot contain any markup.', array('%field' => $form['name']['#title']))); } } /** * Process the language editing form submission. */ function language_admin_edit_form_submit($form, &$form_state) { // Prepare a language object for saving. $languages = language_list(); $langcode = $form_state['values']['langcode']; $language = $languages[$langcode]; $language->name = $form_state['values']['name']; $language->direction = $form_state['values']['direction']; language_save($language); $form_state['redirect'] = 'admin/config/regional/language'; } /** * User interface for the language deletion confirmation screen. */ function language_admin_delete_form($form, &$form_state, $language) { $langcode = $language->id; if (language_default()->id == $langcode) { drupal_set_message(t('The default language cannot be deleted.')); return new RedirectResponse(url('admin/config/regional/language', array('absolute' => TRUE))); } // For other languages, warn the user that data loss is ahead. $languages = language_list(); if (!isset($languages[$langcode])) { throw new NotFoundHttpException(); } else { $form['langcode'] = array('#type' => 'value', '#value' => $langcode); return confirm_form($form, t('Are you sure you want to delete the language %language?', array('%language' => $languages[$langcode]->name)), 'admin/config/regional/language', t('Deleting a language will remove all interface translations associated with it, and posts in this language will be set to be language neutral. This action cannot be undone.'), t('Delete'), t('Cancel')); } } /** * Process language deletion submissions. */ function language_admin_delete_form_submit($form, &$form_state) { $langcode = $form_state['values']['langcode']; $languages = language_list(); $language = $languages[$langcode]; $success = language_delete($langcode); if ($success) { $t_args = array('%language' => $language->name, '%langcode' => $language->id); drupal_set_message(t('The %language (%langcode) language has been removed.', $t_args)); } $form_state['redirect'] = 'admin/config/regional/language'; } /** * Prepare a language code list for unused predefined languages. */ function language_admin_predefined_list() { $languages = language_list(); $predefined = LanguageManager::getStandardLanguageList(); foreach ($predefined as $key => $value) { if (isset($languages[$key])) { unset($predefined[$key]); continue; } $predefined[$key] = t($value[0]); } asort($predefined); return $predefined; } /** * Builds the configuration form for language negotiation. */ function language_negotiation_configure_form() { language_negotiation_include(); $form = array( '#submit' => array('language_negotiation_configure_form_submit'), '#theme' => 'language_negotiation_configure_form', '#language_types' => language_types_get_configurable(FALSE), '#language_types_info' => language_types_info(), '#language_negotiation_info' => language_negotiation_info(), ); foreach ($form['#language_types'] as $type) { language_negotiation_configure_form_table($form, $type); } $form['actions'] = array('#type' => 'actions'); $form['actions']['submit'] = array( '#type' => 'submit', '#value' => t('Save settings'), ); return $form; } /** * Builds a language negotiation method configuration table. */ function language_negotiation_configure_form_table(&$form, $type) { $info = $form['#language_types_info'][$type]; $table_form = array( '#title' => t('@type language detection', array('@type' => $info['name'])), '#tree' => TRUE, '#description' => $info['description'], '#language_negotiation_info' => array(), '#show_operations' => FALSE, 'weight' => array('#tree' => TRUE), ); $negotiation_info = $form['#language_negotiation_info']; $enabled_methods = variable_get("language_negotiation_$type", array()); $methods_weight = variable_get("language_negotiation_methods_weight_$type", array()); // Add missing data to the methods lists. foreach ($negotiation_info as $method_id => $method) { if (!isset($methods_weight[$method_id])) { $methods_weight[$method_id] = isset($method['weight']) ? $method['weight'] : 0; } } // Order methods list by weight. asort($methods_weight); foreach ($methods_weight as $method_id => $weight) { // A language method might be no more available if the defining module has // been disabled after the last configuration saving. if (!isset($negotiation_info[$method_id])) { continue; } $enabled = isset($enabled_methods[$method_id]); $method = $negotiation_info[$method_id]; // List the method only if the current type is defined in its 'types' key. // If it is not defined default to all the configurable language types. $types = array_flip(isset($method['types']) ? $method['types'] : $form['#language_types']); if (isset($types[$type])) { $table_form['#language_negotiation_info'][$method_id] = $method; $method_name = check_plain($method['name']); $table_form['weight'][$method_id] = array( '#type' => 'weight', '#title' => t('Weight for !title language detection method', array('!title' => drupal_strtolower($method_name))), '#title_display' => 'invisible', '#default_value' => $weight, '#attributes' => array('class' => array("language-method-weight-$type")), '#delta' => 20, ); $table_form['title'][$method_id] = array('#markup' => $method_name); $table_form['enabled'][$method_id] = array( '#type' => 'checkbox', '#title' => t('Enable !title language detection method', array('!title' => drupal_strtolower($method_name))), '#title_display' => 'invisible', '#default_value' => $enabled, ); if ($method_id === LANGUAGE_NEGOTIATION_SELECTED) { $table_form['enabled'][$method_id]['#default_value'] = TRUE; $table_form['enabled'][$method_id]['#attributes'] = array('disabled' => 'disabled'); } $table_form['description'][$method_id] = array('#markup' => filter_xss_admin($method['description'])); $config_op = array(); if (isset($method['config'])) { $config_op['configure'] = array( 'title' => t('Configure'), 'href' => $method['config'], ); // If there is at least one operation enabled show the operation column. $table_form['#show_operations'] = TRUE; } $table_form['operation'][$method_id] = array( '#type' => 'operations', '#links' => $config_op, ); } } $form[$type] = $table_form; } /** * Returns HTML for the language negotiation configuration form. * * @param $variables * An associative array containing: * - form: A render element representing the form. * * @ingroup themeable */ function theme_language_negotiation_configure_form($variables) { $form = $variables['form']; $output = ''; foreach ($form['#language_types'] as $type) { $rows = array(); $title = '