'textfield', '#title' => t('Title'), '#default_value' => variable_get('faq_title', 'Frequently Asked Questions'), ); $faq_description_default = array('value' => '', 'format' => filter_fallback_format()); $faq_description = variable_get('faq_description', $faq_description_default); $form['faq_description'] = array( '#type' => 'text_format', '#title' => t('FAQ Description'), '#default_value' => $faq_description['value'], '#description' => t('Your FAQ description. This will be placed at the top of the page, above the questions and can serve as an introductory text.'), '#rows' => 5, '#format' => $faq_description['format'], ); $form['faq_custom_breadcrumbs'] = array( '#type' => 'checkbox', '#title' => t('Create custom breadcrumbs for the FAQ'), '#description' => t('This option set the breadcrumb path to "%home > %faqtitle > category trail".', array( '%home' => t('Home'), '%faqtitle' => variable_get('faq_title', 'Frequently Asked Questions'), ) ), '#default_value' => variable_get('faq_custom_breadcrumbs', TRUE), ); $form['faq_path'] = array( '#type' => 'textfield', '#title' => t('FAQ Path'), '#description' => t('This option sets the path to the faq page. DO NOT append with a \'/\''), '#default_value' => _faq_path(), ); return system_settings_form($form); } /** * Validate for general settings form. */ function faq_general_settings_form_validate($form, &$form_state) { $path = $form_state['values']['faq_path']; // Check if there is no special characters and trailing slash. $pattern = "/[\w-\/]+(? 'hidden', '#value' => variable_get('faq_category_display', 'categories_inline'), ); $form['faq_use_categories'] = array( '#type' => 'hidden', '#value' => variable_get('faq_use_categories', FALSE), ); $display_options['questions_inline'] = t('Questions inline'); $display_options['questions_top'] = t('Clicking on question takes user to answer further down the page'); $display_options['hide_answer'] = t('Clicking on question opens/hides answer under question'); $display_options['new_page'] = t('Clicking on question opens the answer in a new page'); $form['faq_display'] = array( '#type' => 'radios', '#options' => $display_options, '#title' => t('Page layout'), '#description' => t('This controls how the questions and answers are displayed on the page and what happens when someone clicks on the question.'), '#default_value' => variable_get('faq_display', 'questions_top'), ); $form['faq_questions_misc'] = array( '#type' => 'fieldset', '#title' => t('Miscellaneous layout settings'), '#collapsible' => TRUE, ); $form['faq_questions_misc']['faq_question_listing'] = array( '#type' => 'select', '#options' => array( 'ol' => t('Ordered list'), 'ul' => t('Unordered list'), ), '#title' => t('Questions listing style'), '#default_value' => variable_get('faq_question_listing', 'ul'), '#states' => array( 'visible' => array( ':input[name="faq_display"], dummy-0' => array('!value' => 'hide_answer'), ':input[name="faq_display"], dummy-1' => array('!value' => 'questions_inline'), ), ), ); $form['faq_questions_misc']['faq_qa_mark'] = array( '#type' => 'checkbox', '#title' => t('Label questions and answers'), '#description' => t('It labels all questions on the faq page with the "question label" setting and all answers with the "answer label" setting. For example these could be set to "Q:" and "A:".'), '#default_value' => variable_get('faq_qa_mark', FALSE), '#states' => array( 'visible' => array( ':input[name="faq_display"], dummy-0' => array('!value' => 'hide_answer'), ':input[name="faq_display"], dummy-1' => array('!value' => 'new_page'), ), ), ); $form['faq_questions_misc']['faq_question_label'] = array( '#type' => 'textfield', '#title' => t('Question Label'), '#default_value' => variable_get('faq_question_label', 'Q:'), '#states' => array( 'visible' => array( ':input[name="faq_display"], dummy-0' => array('!value' => 'hide_answer'), ':input[name="faq_display"], dummy-1' => array('!value' => 'new_page'), ':input[name="faq_qa_mark"]' => array('checked' => TRUE), ), ), ); $form['faq_questions_misc']['faq_answer_label'] = array( '#type' => 'textfield', '#title' => t('Answer Label'), '#default_value' => variable_get('faq_answer_label', 'A:'), '#states' => array( 'visible' => array( ':input[name="faq_display"], dummy-0' => array('!value' => 'hide_answer'), ':input[name="faq_display"], dummy-1' => array('!value' => 'new_page'), ':input[name="faq_qa_mark"]' => array('checked' => TRUE), ), ), ); $form['faq_questions_misc']['faq_question_long_form'] = array( '#type' => 'checkbox', '#title' => t('Allow long question text to be configured'), '#default_value' => variable_get('faq_question_long_form', 1), ); $form['faq_questions_misc']['faq_question_length'] = array( '#type' => 'radios', '#title' => t('Question length'), '#options' => array( 'long' => t('Display longer text'), 'short' => t('Display short text'), 'both' => t('Display both short and long questions'), ), '#description' => t('The length of question text to display on the FAQ page. The short question will always be displayed in the FAQ blocks.'), '#default_value' => variable_get('faq_question_length', 'short'), '#states' => array( 'visible' => array( ':input[name="faq_question_long_form"]' => array('checked' => TRUE), ), ), ); $form['faq_questions_misc']['faq_hide_qa_accordion'] = array( '#type' => 'checkbox', '#title' => t('Use accordion effect'), '#description' => t('This enables an "accordion" style effect where when a question is clicked, the answer appears beneath, and is then hidden when another question is opened.'), '#default_value' => variable_get('faq_hide_qa_accordion', FALSE), '#states' => array( 'visible' => array( ':input[name="faq_display"]' => array('value' => 'hide_answer'), ), ), ); $form['faq_questions_misc']['faq_show_expand_all'] = array( '#type' => 'checkbox', '#title' => t('Show "expand / collapse all" links for collapsed questions'), '#default_value' => variable_get('faq_show_expand_all', FALSE), '#states' => array( 'invisible' => array( ':input[name="faq_display"]' => array('!value' => 'hide_answer'), ':input[name="faq_use_categories"]' => array('checked' => FALSE), ':input[name="faq_category_display"]' => array('!value' => 'hide_qa'), ), ), ); $form['faq_questions_misc']['faq_use_teaser'] = array( '#type' => 'checkbox', '#title' => t('Use answer teaser'), '#default_value' => variable_get('faq_use_teaser', FALSE), '#states' => array( 'invisible' => array( ':input[name="faq_display"]' => array('value' => 'new_page'), ), ), ); $form['faq_questions_misc']['faq_show_node_links'] = array( '#type' => 'checkbox', '#title' => t('Show node links'), '#description' => t('This enables the display of links under the answer text on the faq page. Examples of these links include "Read more", "Add comment".'), '#default_value' => variable_get('faq_show_node_links', FALSE), '#states' => array( 'invisible' => array( ':input[name="faq_display"]' => array('value' => 'new_page'), ), ), ); $form['faq_questions_misc']['faq_back_to_top'] = array( '#type' => 'textfield', '#title' => t('"Back to Top" link text'), '#description' => t('This allows the user to change the text displayed for the links which return the user to the top of the page. Defaults to "Back to Top". Leave blank to have no link.'), '#default_value' => variable_get('faq_back_to_top', 'Back to Top'), '#states' => array( 'invisible' => array( ':input[name="faq_display"]' => array('value' => 'new_page'), ), ), ); $form['faq_questions_misc']['faq_disable_node_links'] = array( '#type' => 'checkbox', '#title' => t('Disable question links to nodes'), '#description' => t('This allows the user to prevent the questions being links to the faq node.'), '#default_value' => variable_get('faq_disable_node_links', FALSE), '#states' => array( 'invisible' => array( ':input[name="faq_display"]' => array('value' => 'new_page'), ), ), ); $form['faq_questions_misc']['faq_default_sorting'] = array( '#type' => 'select', '#title' => t('Default sorting for unordered FAQs'), '#options' => array( 'DESC' => t('Date Descending'), 'ASC' => t('Date Ascending'), ), '#description' => t("This controls the default ordering behaviour for new FAQ nodes which haven't been assigned a position."), '#default_value' => variable_get('faq_default_sorting', 'DESC'), ); return system_settings_form($form); } /** * Define the elements for the FAQ Settings page - categories tab. * * @return array * The form code inside the $form array. */ function faq_categories_settings_form($form, &$form_state) { $categorization_enabled = variable_get('faq_use_categories', FALSE); $vocabularies = array(); $vocab_omit = array(); $vocab_options = array(); // Display error messages if the taxonomy module isn't enabled or if there is // no valid FAQ vocabulary. if (!module_exists("taxonomy")) { drupal_set_message(t('Categorization of questions will not work without the "taxonomy" module being enabled.'), 'error'); } else { $vocabularies = taxonomy_get_vocabularies('faq'); $vocab_omit = variable_get('faq_omit_vocabulary', array()); foreach ($vocabularies as $vid => $vobj) { $vocab_options[$vid] = $vobj->name; } } // Set up a hidden variable. $form['faq_display'] = array( '#type' => 'hidden', '#value' => variable_get('faq_display', 'questions_top'), ); $form['faq_use_categories'] = array( '#type' => 'checkbox', '#title' => t('Categorize questions'), '#description' => t('This allows the user to display the questions according to the categories configured on the add/edit FAQ page. Use of sub-categories is only recommended for large lists of questions. The Taxonomy module must be enabled.'), '#default_value' => $categorization_enabled, ); $category_options['none'] = t("Don't display"); $category_options['categories_inline'] = t('Categories inline'); $category_options['hide_qa'] = t('Clicking on category opens/hides questions and answers under category'); $category_options['new_page'] = t('Clicking on category opens the questions/answers in a new page'); $form['faq_category_display'] = array( '#type' => 'radios', '#options' => $category_options, '#title' => t('Categories layout'), '#description' => t('This controls how the categories are displayed on the page and what happens when someone clicks on the category.'), '#default_value' => variable_get('faq_category_display', 'categories_inline'), '#states' => array( 'visible' => array( ':input[name="faq_use_categories"]' => array('checked' => TRUE), ), ), ); $form['faq_category_misc'] = array( '#type' => 'fieldset', '#title' => t('Miscellaneous layout settings'), '#collapsible' => TRUE, '#states' => array( 'visible' => array( ':input[name="faq_use_categories"]' => array('checked' => TRUE), ), ), ); $form['faq_category_misc']['faq_category_listing'] = array( '#type' => 'select', '#options' => array( 'ol' => t('Ordered list'), 'ul' => t('Unordered list'), ), '#title' => t('Categories listing style'), '#description' => t('This allows to select how the categories listing is presented. An ordered listing would number the categories, whereas an unordered list will have a bullet to the left of each category.'), '#default_value' => variable_get('faq_category_listing', 'ul'), '#states' => array( 'visible' => array( ':input[name="faq_category_display"]' => array('value' => 'new_page'), ), ), ); $form['faq_category_misc']['faq_category_hide_qa_accordion'] = array( '#type' => 'checkbox', '#title' => t('Use accordion effect'), '#description' => t('This enables an "accordion" style effect where when a category is clicked, the questions appears beneath, and is then hidden when another category is opened.'), '#default_value' => variable_get('faq_category_hide_qa_accordion', FALSE), '#states' => array( 'visible' => array( ':input[name="faq_category_display"]' => array('value' => 'hide_qa'), ), ), ); $form['faq_category_misc']['faq_count'] = array( '#type' => 'checkbox', '#title' => t('Show FAQ count'), '#description' => t('This displays the number of questions in a category after the category name.'), '#default_value' => variable_get('faq_count', FALSE), ); $form['faq_category_misc']['faq_answer_category_name'] = array( '#type' => 'checkbox', '#title' => t('Display category name for answers'), '#description' => t('This allows the user to toggle the visibility of the category name above each answer section.'), '#default_value' => variable_get('faq_answer_category_name', FALSE), '#states' => array( 'visible' => array( ':input[name="faq_display"]' => array('value' => 'hide_qa'), ), ), ); $form['faq_category_misc']['faq_group_questions_top'] = array( '#type' => 'checkbox', '#title' => t("Group questions and answers"), '#default_value' => variable_get('faq_group_questions_top', FALSE), '#states' => array( 'visible' => array( ':input[name="faq_category_display"]' => array('value' => 'categories_inline'), ':input[name="faq_display"]' => array('value' => 'hide_qa'), ), ), ); $form['faq_category_misc']['faq_hide_child_terms'] = array( '#type' => 'checkbox', '#title' => t('Only show sub-categories when parent category is selected'), '#description' => t('This allows the user more control over how and when sub-categories are displayed.'), '#default_value' => variable_get('faq_hide_child_terms', FALSE), '#states' => array( 'invisible' => array( ':input[name="faq_category_display"]' => array('value' => 'categories_inline'), ), ), ); $form['faq_category_misc']['faq_show_term_page_children'] = array( '#type' => 'checkbox', '#title' => t('Show sub-categories on FAQ category pages'), '#description' => t("Sub-categories with 'faq' nodes will be displayed on the per category FAQ page. This will also happen if 'Only show sub-categories when parent category is selected' is set."), '#default_value' => variable_get('faq_show_term_page_children', FALSE), '#states' => array( 'visible' => array( ':input[name="faq_hide_child_terms"]' => array('checked' => FALSE), ':input[name="faq_category_display"]' => array('!value' => 'categories_inline'), ), ), ); if (module_exists('taxonomy') && !empty($vocab_options)) { $form['faq_category_advanced'] = array( '#type' => 'fieldset', '#title' => t('Advanced category settings'), '#collapsible' => TRUE, '#collapsed' => TRUE, '#states' => array( 'visible' => array( ':input[name="faq_use_categories"]' => array('checked' => TRUE), ), ), ); $form['faq_category_advanced']['faq_omit_vocabulary'] = array( '#type' => 'checkboxes', '#title' => t('Omit vocabulary'), '#description' => t('Terms from these vocabularies will be excluded from the FAQ pages.'), '#default_value' => variable_get('faq_omit_vocabulary', array()), '#options' => $vocab_options, '#multiple' => TRUE, ); } $form['#submit'][] = 'faq_categories_settings_form_submit'; return system_settings_form($form); } /** * Display an error if no valid FAQ vocabularies available. */ function faq_categories_settings_form_submit($form, &$form_state) { if ($form_state['values']['faq_use_categories'] && isset($form_state['values']['faq_omit_vocabulary'])) { $vocabularies = taxonomy_get_vocabularies('faq'); $vocab_omit = $form_state['values']['faq_omit_vocabulary']; $valid_vocab = FALSE; foreach ($vocabularies as $vid => $vobj) { if (empty($vocab_omit[$vid])) { $valid_vocab = TRUE; } } if (!$valid_vocab) { drupal_set_message(t('Categories are enabled but no vocabulary is associated with the FAQ content type. Either create a vocabulary or disable categorization in order for questions to appear.'), 'error'); } } } /** * Define the elements for the FAQ Settings page - order tab. * * @param array $form_state * Store the submitted form values. * * @return array * The form code, before being converted to HTML format. */ function faq_order_settings_form($form, $form_state, $category = NULL) { $order = $date_order = ''; drupal_add_js(array('faq' => array('faq_hide_qa_accordion' => variable_get('faq_hide_qa_accordion', FALSE))), array('type' => 'setting', 'scope' => JS_DEFAULT)); drupal_add_js(array('faq' => array('faq_category_hide_qa_accordion' => variable_get('faq_category_hide_qa_accordion', FALSE))), array('type' => 'setting', 'scope' => JS_DEFAULT)); drupal_add_js(drupal_get_path('module', 'faq') . '/faq.js'); drupal_add_css(drupal_get_path('module', 'faq') . '/faq.css'); $use_categories = variable_get('faq_use_categories', FALSE); if (!$use_categories) { $step = "order"; } elseif (!isset($form_state['values']) && empty($category)) { $step = "categories"; } else { $step = "order"; } $form['step'] = array( '#type' => 'value', '#value' => $step, ); // Categorized q/a. if ($step == "categories") { // Get list of categories. $vocabularies = taxonomy_get_vocabularies('faq'); $options = array(); foreach ($vocabularies as $vid => $vobj) { $tree = taxonomy_get_tree($vid); foreach ($tree as $term) { if (!faq_taxonomy_term_count_nodes($term->tid)) { continue; } $options[$term->tid] = faq_tt("taxonomy:term:$term->tid:name", $term->name); $form['choose_cat']['faq_category'] = array( '#type' => 'select', '#title' => t('Choose a category'), '#description' => t('Choose a category that you wish to order the questions for.'), '#options' => $options, '#multiple' => FALSE, ); $form['choose_cat']['search'] = array( '#type' => 'submit', '#value' => t('Search'), '#submit' => array('faq_order_settings_choose_cat_form_submit'), ); } } } else { $default_sorting = variable_get('faq_default_sorting', 'DESC'); $default_weight = 0; if ($default_sorting != 'DESC') { $default_weight = 1000000; } $options = array(); if (!empty($form_state['values']['faq_category'])) { $category = $form_state['values']['faq_category']; } // Uncategorized ordering. $query = db_select('node', 'n') ->fields('n', array('nid', 'title')) ->addTag('node_access') ->condition('n.type', 'faq') ->condition('n.status', 1); // Works, but involves variable concatenation - safe though, since // $default_weight is an integer. $query->addExpression("COALESCE(w.weight, $default_weight)", 'effective_weight'); // Doesn't work in Postgres. // $query->addExpression('COALESCE(w.weight, CAST(:default_weight as SIGNED))', 'effective_weight', array(':default_weight' => $default_weight)); if (empty($category)) { $category = 0; $w_alias = $query->leftJoin('faq_weights', 'w', 'n.nid = %alias.nid AND %alias.tid = :category', array(':category' => $category)); $query->orderBy('effective_weight', 'ASC') ->orderBy('n.sticky', 'DESC') ->orderBy('n.created', $default_sorting == 'DESC' ? 'DESC' : 'ASC'); } // Categorized ordering. else { $ti_alias = $query->innerJoin('taxonomy_index', 'ti', '(n.nid = %alias.nid)'); $w_alias = $query->leftJoin('faq_weights', 'w', 'n.nid = %alias.nid AND %alias.tid = :category', array(':category' => $category)); $query->condition('ti.tid', $category); $query->orderBy('effective_weight', 'ASC') ->orderBy('n.sticky', 'DESC') ->orderBy('n.created', $default_sorting == 'DESC' ? 'DESC' : 'ASC'); } $options = $query->execute()->fetchAll(); $form['weight']['faq_category'] = array( '#type' => 'value', '#value' => $category, ); // Show table ordering form. $form['order_no_cats']['#tree'] = TRUE; $form['order_no_cats']['#theme'] = 'faq_draggable_question_order_table'; $i = 0; foreach ($options as $record) { $form['order_no_cats'][$i]['nid'] = array( '#type' => 'hidden', '#value' => $record->nid, ); $form['order_no_cats'][$i]['title'] = array('#markup' => check_plain($record->title)); $form['order_no_cats'][$i]['sort'] = array( '#type' => 'weight', '#delta' => count($options), '#default_value' => $i, ); $i++; } $form['submit'] = array( '#type' => 'submit', '#value' => t('Save order'), '#weight' => 3, '#submit' => array('faq_order_settings_reorder_form_submit'), ); } return $form; } /** * Function set the rebuild of the form in the FAQ Settings - Weight tab. * * @param array $form * Array, containing the form structure. * * @param array &$form_state * The 'rebuild' key inside $form_state['rebuild'] structure, overrides the * 'redirect' key: when it is set to TRUE, the form will be rebuilt from * scratch and displayed on screen. */ function faq_order_settings_choose_cat_form_submit($form, &$form_state) { $form_state['rebuild'] = TRUE; } /** * Save the options set by the user in the FAQ Settings - Weight tab. * * @param array $form * Array, containing the form structure. * * @param array &$form_state * $form_state['values'] stores the submitted values from the form. */ function faq_order_settings_reorder_form_submit($form, &$form_state) { if ($form_state['values']['op'] == t('Save order') && !empty($form_state['values']['order_no_cats'])) { foreach ($form_state['values']['order_no_cats'] as $i => $faq) { $nid = $faq['nid']; $index = $faq['sort']; db_merge('faq_weights') ->fields(array( 'weight' => $index, )) ->key(array( 'tid' => $form_state['values']['faq_category'], 'nid' => $nid, )) ->execute(); $result = NULL; // If node translation module enabled, update order of the translation // node counterparts. if (module_exists('translation')) { $node = node_load($nid); if ($node->tnid) { $translations = translation_node_get_translations($node->tnid); if (!empty($translations)) { foreach ($translations as $language => $tnode) { db_merge('faq_weights') ->fields(array( 'weight' => $index, )) ->key(array( 'tid' => $form_state['values']['faq_category'], 'nid' => $tnode->nid, )) ->execute(); } } } } } drupal_set_message(t('Configuration has been updated.')); } }