annotate sites/all/modules/webform/components/select.inc @ 0:ff03f76ab3fe

initial version
author danieleb <danielebarchiesi@me.com>
date Wed, 21 Aug 2013 18:51:11 +0100
parents
children
rev   line source
danielebarchiesi@0 1 <?php
danielebarchiesi@0 2
danielebarchiesi@0 3 /**
danielebarchiesi@0 4 * @file
danielebarchiesi@0 5 * Webform module multiple select component.
danielebarchiesi@0 6 */
danielebarchiesi@0 7
danielebarchiesi@0 8 /**
danielebarchiesi@0 9 * Implements _webform_defaults_component().
danielebarchiesi@0 10 */
danielebarchiesi@0 11 function _webform_defaults_select() {
danielebarchiesi@0 12 return array(
danielebarchiesi@0 13 'name' => '',
danielebarchiesi@0 14 'form_key' => NULL,
danielebarchiesi@0 15 'mandatory' => 0,
danielebarchiesi@0 16 'pid' => 0,
danielebarchiesi@0 17 'weight' => 0,
danielebarchiesi@0 18 'value' => '',
danielebarchiesi@0 19 'extra' => array(
danielebarchiesi@0 20 'items' => '',
danielebarchiesi@0 21 'multiple' => NULL,
danielebarchiesi@0 22 'aslist' => NULL,
danielebarchiesi@0 23 'optrand' => 0,
danielebarchiesi@0 24 'other_option' => NULL,
danielebarchiesi@0 25 'other_text' => t('Other...'),
danielebarchiesi@0 26 'title_display' => 0,
danielebarchiesi@0 27 'description' => '',
danielebarchiesi@0 28 'custom_keys' => FALSE,
danielebarchiesi@0 29 'options_source' => '',
danielebarchiesi@0 30 'private' => FALSE,
danielebarchiesi@0 31 ),
danielebarchiesi@0 32 );
danielebarchiesi@0 33 }
danielebarchiesi@0 34
danielebarchiesi@0 35 /**
danielebarchiesi@0 36 * Implements _webform_theme_component().
danielebarchiesi@0 37 */
danielebarchiesi@0 38 function _webform_theme_select() {
danielebarchiesi@0 39 return array(
danielebarchiesi@0 40 'webform_display_select' => array(
danielebarchiesi@0 41 'render element' => 'element',
danielebarchiesi@0 42 'file' => 'components/select.inc',
danielebarchiesi@0 43 ),
danielebarchiesi@0 44 );
danielebarchiesi@0 45 }
danielebarchiesi@0 46
danielebarchiesi@0 47 /**
danielebarchiesi@0 48 * Implements _webform_edit_component().
danielebarchiesi@0 49 */
danielebarchiesi@0 50 function _webform_edit_select($component) {
danielebarchiesi@0 51 $form = array(
danielebarchiesi@0 52 '#attached' => array(
danielebarchiesi@0 53 'js' => array(
danielebarchiesi@0 54 drupal_get_path('module', 'webform') . '/js/select-admin.js' => array('preprocess' => FALSE),
danielebarchiesi@0 55 array('data' => array('webform' => array('selectOptionsUrl' => url('webform/ajax/options/' . $component['nid']))), 'type' => 'setting'),
danielebarchiesi@0 56 ),
danielebarchiesi@0 57 ),
danielebarchiesi@0 58 );
danielebarchiesi@0 59
danielebarchiesi@0 60 $other = array();
danielebarchiesi@0 61 if ($info = _webform_select_options_info()) {
danielebarchiesi@0 62 $options = array('' => t('None'));
danielebarchiesi@0 63 foreach ($info as $name => $source) {
danielebarchiesi@0 64 $options[$name] = $source['title'];
danielebarchiesi@0 65 }
danielebarchiesi@0 66
danielebarchiesi@0 67 $other['options_source'] = array(
danielebarchiesi@0 68 '#title' => t('Load a pre-built option list'),
danielebarchiesi@0 69 '#type' => 'select',
danielebarchiesi@0 70 '#options' => $options,
danielebarchiesi@0 71 '#default_value' => $component['extra']['options_source'],
danielebarchiesi@0 72 '#weight' => 1,
danielebarchiesi@0 73 '#description' => t('Use a pre-built list of options rather than entering options manually. Options will not be editable if using pre-built list.'),
danielebarchiesi@0 74 '#parents' => array('extra', 'options_source'),
danielebarchiesi@0 75 '#weight' => 5,
danielebarchiesi@0 76 );
danielebarchiesi@0 77 }
danielebarchiesi@0 78
danielebarchiesi@0 79 if (module_exists('select_or_other')) {
danielebarchiesi@0 80 $other['other_option'] = array(
danielebarchiesi@0 81 '#type' => 'checkbox',
danielebarchiesi@0 82 '#title' => t('Allow "Other..." option'),
danielebarchiesi@0 83 '#default_value' => $component['extra']['other_option'],
danielebarchiesi@0 84 '#description' => t('Check this option if you want to allow users to enter an option not on the list.'),
danielebarchiesi@0 85 '#parents' => array('extra', 'other_option'),
danielebarchiesi@0 86 '#weight' => 2,
danielebarchiesi@0 87 );
danielebarchiesi@0 88 $other['other_text'] = array(
danielebarchiesi@0 89 '#type' => 'textfield',
danielebarchiesi@0 90 '#title' => t('Text for "Other..." option'),
danielebarchiesi@0 91 '#default_value' => $component['extra']['other_text'],
danielebarchiesi@0 92 '#description' => t('If allowing other options, enter text to be used for other-enabling option.'),
danielebarchiesi@0 93 '#parents' => array('extra', 'other_text'),
danielebarchiesi@0 94 '#weight' => 3,
danielebarchiesi@0 95 );
danielebarchiesi@0 96 }
danielebarchiesi@0 97
danielebarchiesi@0 98 if (module_exists('options_element')) {
danielebarchiesi@0 99 $options = _webform_select_options($component, FALSE, FALSE);
danielebarchiesi@0 100
danielebarchiesi@0 101 $form['items'] = array(
danielebarchiesi@0 102 '#type' => 'fieldset',
danielebarchiesi@0 103 '#title' => t('Options'),
danielebarchiesi@0 104 '#collapsible' => TRUE,
danielebarchiesi@0 105 '#attributes' => array('class' => array('webform-options-element')),
danielebarchiesi@0 106 '#element_validate' => array('_webform_edit_validate_options'),
danielebarchiesi@0 107 '#weight' => 2,
danielebarchiesi@0 108 );
danielebarchiesi@0 109
danielebarchiesi@0 110 $form['items']['options'] = array(
danielebarchiesi@0 111 '#type' => 'options',
danielebarchiesi@0 112 '#limit' => 500,
danielebarchiesi@0 113 '#optgroups' => $component['extra']['aslist'],
danielebarchiesi@0 114 '#multiple' => $component['extra']['multiple'],
danielebarchiesi@0 115 '#multiple_toggle' => t('Multiple'),
danielebarchiesi@0 116 '#default_value' => $component['value'],
danielebarchiesi@0 117 '#options' => $options,
danielebarchiesi@0 118 '#options_readonly' => !empty($component['extra']['options_source']),
danielebarchiesi@0 119 '#key_type' => 'mixed',
danielebarchiesi@0 120 '#key_type_toggle' => t('Customize keys (Advanced)'),
danielebarchiesi@0 121 '#key_type_toggled' => $component['extra']['custom_keys'],
danielebarchiesi@0 122 '#default_value_pattern' => '^%.+\[.+\]$',
danielebarchiesi@0 123 '#weight' => 1,
danielebarchiesi@0 124 );
danielebarchiesi@0 125
danielebarchiesi@0 126 $form['items']['options']['option_settings'] = $other;
danielebarchiesi@0 127 }
danielebarchiesi@0 128 else {
danielebarchiesi@0 129 $form['extra']['items'] = array(
danielebarchiesi@0 130 '#type' => 'textarea',
danielebarchiesi@0 131 '#title' => t('Options'),
danielebarchiesi@0 132 '#default_value' => $component['extra']['items'],
danielebarchiesi@0 133 '#description' => t('<strong>Key-value pairs MUST be specified as "safe_key|Some readable option"</strong>. Use of only alphanumeric characters and underscores is recommended in keys. One option per line. Option groups may be specified with &lt;Group Name&gt;. &lt;&gt; can be used to insert items at the root of the menu after specifying a group.') . theme('webform_token_help'),
danielebarchiesi@0 134 '#cols' => 60,
danielebarchiesi@0 135 '#rows' => 5,
danielebarchiesi@0 136 '#weight' => 0,
danielebarchiesi@0 137 '#required' => TRUE,
danielebarchiesi@0 138 '#wysiwyg' => FALSE,
danielebarchiesi@0 139 '#element_validate' => array('_webform_edit_validate_select'),
danielebarchiesi@0 140 );
danielebarchiesi@0 141
danielebarchiesi@0 142 if (!empty($component['extra']['options_source'])) {
danielebarchiesi@0 143 $form['extra']['items']['#attributes'] = array('readonly' => 'readonly');
danielebarchiesi@0 144 }
danielebarchiesi@0 145
danielebarchiesi@0 146 $form['extra'] = array_merge($form['extra'], $other);
danielebarchiesi@0 147 $form['value'] = array(
danielebarchiesi@0 148 '#type' => 'textfield',
danielebarchiesi@0 149 '#title' => t('Default value'),
danielebarchiesi@0 150 '#default_value' => $component['value'],
danielebarchiesi@0 151 '#description' => t('The default value of the field identified by its key. For multiple selects use commas to separate multiple defaults.') . theme('webform_token_help'),
danielebarchiesi@0 152 '#size' => 60,
danielebarchiesi@0 153 '#maxlength' => 1024,
danielebarchiesi@0 154 '#weight' => 0,
danielebarchiesi@0 155 );
danielebarchiesi@0 156 $form['extra']['multiple'] = array(
danielebarchiesi@0 157 '#type' => 'checkbox',
danielebarchiesi@0 158 '#title' => t('Multiple'),
danielebarchiesi@0 159 '#default_value' => $component['extra']['multiple'],
danielebarchiesi@0 160 '#description' => t('Check this option if the user should be allowed to choose multiple values.'),
danielebarchiesi@0 161 '#weight' => 0,
danielebarchiesi@0 162 );
danielebarchiesi@0 163 }
danielebarchiesi@0 164
danielebarchiesi@0 165 $form['display']['aslist'] = array(
danielebarchiesi@0 166 '#type' => 'checkbox',
danielebarchiesi@0 167 '#title' => t('Listbox'),
danielebarchiesi@0 168 '#default_value' => $component['extra']['aslist'],
danielebarchiesi@0 169 '#description' => t('Check this option if you want the select component to be displayed as a select list box instead of radio buttons or checkboxes. Option groups (nested options) are only supported with listbox components.'),
danielebarchiesi@0 170 '#parents' => array('extra', 'aslist'),
danielebarchiesi@0 171 );
danielebarchiesi@0 172 $form['display']['optrand'] = array(
danielebarchiesi@0 173 '#type' => 'checkbox',
danielebarchiesi@0 174 '#title' => t('Randomize options'),
danielebarchiesi@0 175 '#default_value' => $component['extra']['optrand'],
danielebarchiesi@0 176 '#description' => t('Randomizes the order of the options when they are displayed in the form.'),
danielebarchiesi@0 177 '#parents' => array('extra', 'optrand'),
danielebarchiesi@0 178 );
danielebarchiesi@0 179
danielebarchiesi@0 180 return $form;
danielebarchiesi@0 181 }
danielebarchiesi@0 182
danielebarchiesi@0 183 /**
danielebarchiesi@0 184 * Element validation callback. Ensure keys are not duplicated.
danielebarchiesi@0 185 */
danielebarchiesi@0 186 function _webform_edit_validate_select($element, &$form_state) {
danielebarchiesi@0 187 // Check for duplicate key values to prevent unexpected data loss. Require
danielebarchiesi@0 188 // all options to include a safe_key.
danielebarchiesi@0 189 if (!empty($element['#value'])) {
danielebarchiesi@0 190 $lines = explode("\n", trim($element['#value']));
danielebarchiesi@0 191 $existing_keys = array();
danielebarchiesi@0 192 $duplicate_keys = array();
danielebarchiesi@0 193 $missing_keys = array();
danielebarchiesi@0 194 $long_keys = array();
danielebarchiesi@0 195 $group = '';
danielebarchiesi@0 196 foreach ($lines as $line) {
danielebarchiesi@0 197 $matches = array();
danielebarchiesi@0 198 $line = trim($line);
danielebarchiesi@0 199 if (preg_match('/^\<([^>]*)\>$/', $line, $matches)) {
danielebarchiesi@0 200 $group = $matches[1];
danielebarchiesi@0 201 $key = NULL; // No need to store group names.
danielebarchiesi@0 202 }
danielebarchiesi@0 203 elseif (preg_match('/^([^|]*)\|(.*)$/', $line, $matches)) {
danielebarchiesi@0 204 $key = $matches[1];
danielebarchiesi@0 205 if (strlen($key) > 128) {
danielebarchiesi@0 206 $long_keys[] = $key;
danielebarchiesi@0 207 }
danielebarchiesi@0 208 }
danielebarchiesi@0 209 else {
danielebarchiesi@0 210 $missing_keys[] = $line;
danielebarchiesi@0 211 }
danielebarchiesi@0 212
danielebarchiesi@0 213 if (isset($key)) {
danielebarchiesi@0 214 if (isset($existing_keys[$group][$key])) {
danielebarchiesi@0 215 $duplicate_keys[$key] = $key;
danielebarchiesi@0 216 }
danielebarchiesi@0 217 else {
danielebarchiesi@0 218 $existing_keys[$group][$key] = $key;
danielebarchiesi@0 219 }
danielebarchiesi@0 220 }
danielebarchiesi@0 221 }
danielebarchiesi@0 222
danielebarchiesi@0 223 if (!empty($missing_keys)) {
danielebarchiesi@0 224 form_error($element, t('Every option must have a key specified. Specify each option as "safe_key|Some readable option".'));
danielebarchiesi@0 225 }
danielebarchiesi@0 226
danielebarchiesi@0 227 if (!empty($long_keys)) {
danielebarchiesi@0 228 form_error($element, t('Option keys must be less than 128 characters. The following keys exceed this limit:') . theme('item_list', $long_keys));
danielebarchiesi@0 229 }
danielebarchiesi@0 230
danielebarchiesi@0 231 if (!empty($duplicate_keys)) {
danielebarchiesi@0 232 form_error($element, t('Options within the select list must be unique. The following keys have been used multiple times:') . theme('item_list', array('items' => $duplicate_keys)));
danielebarchiesi@0 233 }
danielebarchiesi@0 234
danielebarchiesi@0 235 // Set the listbox option if needed.
danielebarchiesi@0 236 if (empty($missing_keys) && empty($long_keys) && empty($duplicate_keys)) {
danielebarchiesi@0 237 $options = _webform_select_options_from_text($element['#value']);
danielebarchiesi@0 238 _webform_edit_validate_set_aslist($options, $form_state);
danielebarchiesi@0 239 }
danielebarchiesi@0 240 }
danielebarchiesi@0 241
danielebarchiesi@0 242 return TRUE;
danielebarchiesi@0 243 }
danielebarchiesi@0 244
danielebarchiesi@0 245 /**
danielebarchiesi@0 246 * Set the appropriate webform values when using the options element module.
danielebarchiesi@0 247 */
danielebarchiesi@0 248 function _webform_edit_validate_options($element, &$form_state) {
danielebarchiesi@0 249 $key = end($element['#parents']);
danielebarchiesi@0 250 $element_options = $form_state['values'][$key]['options'];
danielebarchiesi@0 251 unset($form_state['values'][$key]);
danielebarchiesi@0 252
danielebarchiesi@0 253 $form_state['values']['extra'][$key] = form_options_to_text($element_options['options'], 'custom');
danielebarchiesi@0 254
danielebarchiesi@0 255 // Options saved for select components.
danielebarchiesi@0 256 if ($key == 'items') {
danielebarchiesi@0 257 $form_state['values']['extra']['multiple'] = $element_options['multiple'];
danielebarchiesi@0 258 $form_state['values']['extra']['custom_keys'] = $element_options['custom_keys'];
danielebarchiesi@0 259 $form_state['values']['value'] = is_array($element_options['default_value']) ? implode(', ', $element_options['default_value']) : $element_options['default_value'];
danielebarchiesi@0 260
danielebarchiesi@0 261 // Set the listbox option if needed.
danielebarchiesi@0 262 _webform_edit_validate_set_aslist($element_options['options'], $form_state);
danielebarchiesi@0 263 }
danielebarchiesi@0 264 // Options saved for grid components.
danielebarchiesi@0 265 else {
danielebarchiesi@0 266 $form_state['values']['extra']['custom_' . rtrim($key, 's') . '_keys'] = $element_options['custom_keys'];
danielebarchiesi@0 267 }
danielebarchiesi@0 268 }
danielebarchiesi@0 269
danielebarchiesi@0 270 /**
danielebarchiesi@0 271 * Ensure "aslist" is used for option groups. Called from options validations.
danielebarchiesi@0 272 */
danielebarchiesi@0 273 function _webform_edit_validate_set_aslist($options, &$form_state) {
danielebarchiesi@0 274 if (empty($form_state['values']['extra']['aslist']) && !empty($options)) {
danielebarchiesi@0 275 foreach ($options as $option) {
danielebarchiesi@0 276 if (is_array($option)) {
danielebarchiesi@0 277 $form_state['values']['extra']['aslist'] = 1;
danielebarchiesi@0 278 drupal_set_message(t('The component %name has automatically been set to display as a listbox in order to support option groups.', array('%name' => $form_state['values']['name'])), 'warning');
danielebarchiesi@0 279 break;
danielebarchiesi@0 280 }
danielebarchiesi@0 281 }
danielebarchiesi@0 282 }
danielebarchiesi@0 283 }
danielebarchiesi@0 284
danielebarchiesi@0 285 /**
danielebarchiesi@0 286 * Implements _webform_render_component().
danielebarchiesi@0 287 */
danielebarchiesi@0 288 function _webform_render_select($component, $value = NULL, $filter = TRUE) {
danielebarchiesi@0 289 $node = isset($component['nid']) ? node_load($component['nid']) : NULL;
danielebarchiesi@0 290
danielebarchiesi@0 291 $element = array(
danielebarchiesi@0 292 '#title' => $filter ? _webform_filter_xss($component['name']) : $component['name'],
danielebarchiesi@0 293 '#title_display' => $component['extra']['title_display'] ? $component['extra']['title_display'] : 'before',
danielebarchiesi@0 294 '#required' => $component['mandatory'],
danielebarchiesi@0 295 '#weight' => $component['weight'],
danielebarchiesi@0 296 '#description' => $filter ? _webform_filter_descriptions($component['extra']['description'], $node) : $component['extra']['description'],
danielebarchiesi@0 297 '#theme_wrappers' => array('webform_element'),
danielebarchiesi@0 298 '#pre_render' => array(), // Needed to disable double-wrapping of radios and checkboxes.
danielebarchiesi@0 299 '#translatable' => array('title', 'description', 'options'),
danielebarchiesi@0 300 );
danielebarchiesi@0 301
danielebarchiesi@0 302 // Convert the user-entered options list into an array.
danielebarchiesi@0 303 $default_value = $filter ? _webform_filter_values($component['value'], $node, NULL, NULL, FALSE) : $component['value'];
danielebarchiesi@0 304 $options = _webform_select_options($component, !$component['extra']['aslist'], $filter);
danielebarchiesi@0 305
danielebarchiesi@0 306 if ($component['extra']['optrand']) {
danielebarchiesi@0 307 _webform_shuffle_options($options);
danielebarchiesi@0 308 }
danielebarchiesi@0 309
danielebarchiesi@0 310 // Add default options if using a select list with no default. This trigger's
danielebarchiesi@0 311 // Drupal 7's adding of the option for us. See @form_process_select().
danielebarchiesi@0 312 if ($component['extra']['aslist'] && !$component['extra']['multiple'] && $default_value === '') {
danielebarchiesi@0 313 $element['#empty_value'] = '';
danielebarchiesi@0 314 }
danielebarchiesi@0 315
danielebarchiesi@0 316 // Set the component options.
danielebarchiesi@0 317 $element['#options'] = $options;
danielebarchiesi@0 318
danielebarchiesi@0 319 // Set the default value.
danielebarchiesi@0 320 if (isset($value)) {
danielebarchiesi@0 321 if ($component['extra']['multiple']) {
danielebarchiesi@0 322 // Set the value as an array.
danielebarchiesi@0 323 $element['#default_value'] = array();
danielebarchiesi@0 324 foreach ((array) $value as $key => $option_value) {
danielebarchiesi@0 325 $element['#default_value'][] = $option_value;
danielebarchiesi@0 326 }
danielebarchiesi@0 327 }
danielebarchiesi@0 328 else {
danielebarchiesi@0 329 // Set the value as a single string.
danielebarchiesi@0 330 $element['#default_value'] = '';
danielebarchiesi@0 331 foreach ((array) $value as $option_value) {
danielebarchiesi@0 332 $element['#default_value'] = $option_value;
danielebarchiesi@0 333 }
danielebarchiesi@0 334 }
danielebarchiesi@0 335 }
danielebarchiesi@0 336 elseif ($default_value !== '') {
danielebarchiesi@0 337 // Convert default value to a list if necessary.
danielebarchiesi@0 338 if ($component['extra']['multiple']) {
danielebarchiesi@0 339 $varray = explode(',', $default_value);
danielebarchiesi@0 340 foreach ($varray as $key => $v) {
danielebarchiesi@0 341 $v = trim($v);
danielebarchiesi@0 342 if ($v !== '') {
danielebarchiesi@0 343 $element['#default_value'][] = $v;
danielebarchiesi@0 344 }
danielebarchiesi@0 345 }
danielebarchiesi@0 346 }
danielebarchiesi@0 347 else {
danielebarchiesi@0 348 $element['#default_value'] = $default_value;
danielebarchiesi@0 349 }
danielebarchiesi@0 350 }
danielebarchiesi@0 351 elseif ($component['extra']['multiple']) {
danielebarchiesi@0 352 $element['#default_value'] = array();
danielebarchiesi@0 353 }
danielebarchiesi@0 354
danielebarchiesi@0 355 if ($component['extra']['other_option'] && module_exists('select_or_other')) {
danielebarchiesi@0 356 // Set display as a select_or_other element:
danielebarchiesi@0 357 $element['#type'] = 'select_or_other';
danielebarchiesi@0 358 $element['#other'] = !empty($component['extra']['other_text']) ? check_plain($component['extra']['other_text']) : t('Other...');
danielebarchiesi@0 359 $element['#other_title'] = $element['#title'] . ' ' . $element['#other'];
danielebarchiesi@0 360 $element['#other_title_display'] = 'invisible';
danielebarchiesi@0 361 $element['#other_unknown_defaults'] = 'other';
danielebarchiesi@0 362 $element['#other_delimiter'] = ', ';
danielebarchiesi@0 363 // Merge in Webform's #process function for Select or other.
danielebarchiesi@0 364 $element['#process'] = array_merge(element_info_property('select_or_other', '#process'), array('webform_expand_select_or_other'));
danielebarchiesi@0 365
danielebarchiesi@0 366 if ($component['extra']['multiple']) {
danielebarchiesi@0 367 $element['#multiple'] = TRUE;
danielebarchiesi@0 368 $element['#select_type'] = 'checkboxes';
danielebarchiesi@0 369 }
danielebarchiesi@0 370 else {
danielebarchiesi@0 371 $element['#multiple'] = FALSE;
danielebarchiesi@0 372 $element['#select_type'] = 'radios';
danielebarchiesi@0 373 }
danielebarchiesi@0 374 if ($component['extra']['aslist']) {
danielebarchiesi@0 375 $element['#select_type'] = 'select';
danielebarchiesi@0 376 }
danielebarchiesi@0 377 }
danielebarchiesi@0 378 elseif ($component['extra']['aslist']) {
danielebarchiesi@0 379 // Set display as a select list:
danielebarchiesi@0 380 $element['#type'] = 'select';
danielebarchiesi@0 381 if ($component['extra']['multiple']) {
danielebarchiesi@0 382 $element['#size'] = 4;
danielebarchiesi@0 383 $element['#multiple'] = TRUE;
danielebarchiesi@0 384 }
danielebarchiesi@0 385 }
danielebarchiesi@0 386 else {
danielebarchiesi@0 387 if ($component['extra']['multiple']) {
danielebarchiesi@0 388 // Set display as a checkbox set.
danielebarchiesi@0 389 $element['#type'] = 'checkboxes';
danielebarchiesi@0 390 $element['#theme_wrappers'] = array_merge(array('checkboxes'), $element['#theme_wrappers']);
danielebarchiesi@0 391 $element['#process'] = array_merge(element_info_property('checkboxes', '#process'), array('webform_expand_select_ids'));
danielebarchiesi@0 392
danielebarchiesi@0 393 // Entirely replace the normal expand checkboxes with our custom version.
danielebarchiesi@0 394 // This helps render checkboxes in multipage forms.
danielebarchiesi@0 395 $process_key = array_search('form_process_checkboxes', $element['#process']);
danielebarchiesi@0 396 $element['#process'][$process_key] = 'webform_expand_checkboxes';
danielebarchiesi@0 397 }
danielebarchiesi@0 398 else {
danielebarchiesi@0 399 // Set display as a radio set.
danielebarchiesi@0 400 $element['#type'] = 'radios';
danielebarchiesi@0 401 $element['#theme_wrappers'] = array_merge(array('radios'), $element['#theme_wrappers']);
danielebarchiesi@0 402 $element['#process'] = array_merge(element_info_property('radios', '#process'), array('webform_expand_select_ids'));
danielebarchiesi@0 403 }
danielebarchiesi@0 404 }
danielebarchiesi@0 405
danielebarchiesi@0 406 return $element;
danielebarchiesi@0 407 }
danielebarchiesi@0 408
danielebarchiesi@0 409 /**
danielebarchiesi@0 410 * Process function to ensure select_or_other elements validate properly.
danielebarchiesi@0 411 */
danielebarchiesi@0 412 function webform_expand_select_or_other($element) {
danielebarchiesi@0 413 // Disable validation for back-button and save draft.
danielebarchiesi@0 414 $element['select']['#validated'] = TRUE;
danielebarchiesi@0 415 $element['select']['#webform_validated'] = FALSE;
danielebarchiesi@0 416
danielebarchiesi@0 417 $element['other']['#validated'] = TRUE;
danielebarchiesi@0 418 $element['other']['#webform_validated'] = FALSE;
danielebarchiesi@0 419
danielebarchiesi@0 420 // The Drupal FAPI does not support #title_display inline so we need to move
danielebarchiesi@0 421 // to a supported value here to be compatible with select_or_other.
danielebarchiesi@0 422 $element['select']['#title_display'] = $element['#title_display'] === 'inline' ? 'before' : $element['#title_display'];
danielebarchiesi@0 423
danielebarchiesi@0 424 // If the default value contains "select_or_other" (the key of the select
danielebarchiesi@0 425 // element for the "other..." choice), discard it and set the "other" value.
danielebarchiesi@0 426 if (is_array($element['#default_value']) && in_array('select_or_other', $element['#default_value'])) {
danielebarchiesi@0 427 $key = array_search('select_or_other', $element['#default_value']);
danielebarchiesi@0 428 unset($element['#default_value'][$key]);
danielebarchiesi@0 429 $element['#default_value'] = array_values($element['#default_value']);
danielebarchiesi@0 430 $element['other']['#default_value'] = implode(', ', $element['#default_value']);
danielebarchiesi@0 431 }
danielebarchiesi@0 432
danielebarchiesi@0 433 // Sanitize the options in Select or other check boxes and radio buttons.
danielebarchiesi@0 434 if ($element['#select_type'] == 'checkboxes' || $element['#select_type'] == 'radios') {
danielebarchiesi@0 435 $element['select']['#process'] = array_merge(element_info_property($element['#select_type'], '#process'), array('webform_expand_select_ids'));
danielebarchiesi@0 436 }
danielebarchiesi@0 437
danielebarchiesi@0 438 return $element;
danielebarchiesi@0 439 }
danielebarchiesi@0 440
danielebarchiesi@0 441 /**
danielebarchiesi@0 442 * Drupal 6 hack that properly *renders* checkboxes in multistep forms. This is
danielebarchiesi@0 443 * different than the value hack needed in Drupal 5, which is no longer needed.
danielebarchiesi@0 444 */
danielebarchiesi@0 445 function webform_expand_checkboxes($element) {
danielebarchiesi@0 446 // Elements that have a value set are already in the form structure cause
danielebarchiesi@0 447 // them not to be written when the expand_checkboxes function is called.
danielebarchiesi@0 448 $default_value = array();
danielebarchiesi@0 449 foreach (element_children($element) as $key) {
danielebarchiesi@0 450 if (isset($element[$key]['#default_value'])) {
danielebarchiesi@0 451 $default_value[$key] = $element[$key]['#default_value'];
danielebarchiesi@0 452 unset($element[$key]);
danielebarchiesi@0 453 }
danielebarchiesi@0 454 }
danielebarchiesi@0 455
danielebarchiesi@0 456 $element = form_process_checkboxes($element);
danielebarchiesi@0 457
danielebarchiesi@0 458 // Escape the values of checkboxes.
danielebarchiesi@0 459 foreach (element_children($element) as $key) {
danielebarchiesi@0 460 $element[$key]['#return_value'] = check_plain($element[$key]['#return_value']);
danielebarchiesi@0 461 $element[$key]['#name'] = $element['#name'] . '[' . $element[$key]['#return_value'] . ']';
danielebarchiesi@0 462 }
danielebarchiesi@0 463
danielebarchiesi@0 464 foreach ($default_value as $key => $val) {
danielebarchiesi@0 465 $element[$key]['#default_value'] = $val;
danielebarchiesi@0 466 }
danielebarchiesi@0 467 return $element;
danielebarchiesi@0 468 }
danielebarchiesi@0 469
danielebarchiesi@0 470 /**
danielebarchiesi@0 471 * FAPI process function to rename IDs attached to checkboxes and radios.
danielebarchiesi@0 472 */
danielebarchiesi@0 473 function webform_expand_select_ids($element) {
danielebarchiesi@0 474 $id = $element['#id'] = str_replace('_', '-', _webform_safe_name(strip_tags($element['#id'])));
danielebarchiesi@0 475 $delta = 0;
danielebarchiesi@0 476 foreach (element_children($element) as $key) {
danielebarchiesi@0 477 $delta++;
danielebarchiesi@0 478 // Convert the #id for each child to a safe name, regardless of key.
danielebarchiesi@0 479 $element[$key]['#id'] = $id . '-' . $delta;
danielebarchiesi@0 480
danielebarchiesi@0 481 // Prevent scripts or CSS in the labels for each checkbox or radio.
danielebarchiesi@0 482 $element[$key]['#title'] = _webform_filter_xss($element[$key]['#title']);
danielebarchiesi@0 483 }
danielebarchiesi@0 484 return $element;
danielebarchiesi@0 485 }
danielebarchiesi@0 486
danielebarchiesi@0 487 /**
danielebarchiesi@0 488 * Implements _webform_display_component().
danielebarchiesi@0 489 */
danielebarchiesi@0 490 function _webform_display_select($component, $value, $format = 'html') {
danielebarchiesi@0 491 return array(
danielebarchiesi@0 492 '#title' => $component['name'],
danielebarchiesi@0 493 '#weight' => $component['weight'],
danielebarchiesi@0 494 '#multiple' => $component['extra']['multiple'],
danielebarchiesi@0 495 '#theme' => 'webform_display_select',
danielebarchiesi@0 496 '#theme_wrappers' => $format == 'html' ? array('webform_element') : array('webform_element_text'),
danielebarchiesi@0 497 '#format' => $format,
danielebarchiesi@0 498 '#options' => _webform_select_options($component, !$component['extra']['aslist']),
danielebarchiesi@0 499 '#value' => (array) $value,
danielebarchiesi@0 500 '#translatable' => array('title', 'options'),
danielebarchiesi@0 501 );
danielebarchiesi@0 502 }
danielebarchiesi@0 503
danielebarchiesi@0 504 /**
danielebarchiesi@0 505 * Implements _webform_submit_component().
danielebarchiesi@0 506 *
danielebarchiesi@0 507 * Convert FAPI 0/1 values into something saveable.
danielebarchiesi@0 508 */
danielebarchiesi@0 509 function _webform_submit_select($component, $value) {
danielebarchiesi@0 510 // Build a list of all valid keys expected to be submitted.
danielebarchiesi@0 511 $options = _webform_select_options($component, TRUE);
danielebarchiesi@0 512
danielebarchiesi@0 513 $return = NULL;
danielebarchiesi@0 514 if (is_array($value)) {
danielebarchiesi@0 515 $return = array();
danielebarchiesi@0 516 foreach ($value as $key => $option_value) {
danielebarchiesi@0 517 // Handle options that are specified options.
danielebarchiesi@0 518 if ($option_value !== '' && isset($options[$option_value])) {
danielebarchiesi@0 519 // Checkboxes submit an integer value of 0 when unchecked. A checkbox
danielebarchiesi@0 520 // with a value of '0' is valid, so we can't use empty() here.
danielebarchiesi@0 521 if ($option_value === 0 && !$component['extra']['aslist'] && $component['extra']['multiple']) {
danielebarchiesi@0 522 unset($value[$option_value]);
danielebarchiesi@0 523 }
danielebarchiesi@0 524 else {
danielebarchiesi@0 525 $return[] = $option_value;
danielebarchiesi@0 526 }
danielebarchiesi@0 527 }
danielebarchiesi@0 528 // Handle options that are added through the "other" field. Specifically
danielebarchiesi@0 529 // exclude the "select_or_other" value, which is added by the select list.
danielebarchiesi@0 530 elseif ($component['extra']['other_option'] && module_exists('select_or_other') && $option_value != 'select_or_other') {
danielebarchiesi@0 531 $return[] = $option_value;
danielebarchiesi@0 532 }
danielebarchiesi@0 533 }
danielebarchiesi@0 534 }
danielebarchiesi@0 535 elseif (is_string($value)) {
danielebarchiesi@0 536 $return = $value;
danielebarchiesi@0 537 }
danielebarchiesi@0 538
danielebarchiesi@0 539 return $return;
danielebarchiesi@0 540 }
danielebarchiesi@0 541
danielebarchiesi@0 542 /**
danielebarchiesi@0 543 * Format the text output for this component.
danielebarchiesi@0 544 */
danielebarchiesi@0 545 function theme_webform_display_select($variables) {
danielebarchiesi@0 546 $element = $variables['element'];
danielebarchiesi@0 547
danielebarchiesi@0 548 // Flatten the list of options so we can get values easily. These options
danielebarchiesi@0 549 // may be translated by hook_webform_display_component_alter().
danielebarchiesi@0 550 $options = array();
danielebarchiesi@0 551 foreach ($element['#options'] as $key => $value) {
danielebarchiesi@0 552 if (is_array($value)) {
danielebarchiesi@0 553 foreach ($value as $subkey => $subvalue) {
danielebarchiesi@0 554 $options[$subkey] = $subvalue;
danielebarchiesi@0 555 }
danielebarchiesi@0 556 }
danielebarchiesi@0 557 else {
danielebarchiesi@0 558 $options[$key] = $value;
danielebarchiesi@0 559 }
danielebarchiesi@0 560 }
danielebarchiesi@0 561
danielebarchiesi@0 562 $items = array();
danielebarchiesi@0 563 if ($element['#multiple']) {
danielebarchiesi@0 564 foreach ((array) $element['#value'] as $option_value) {
danielebarchiesi@0 565 if ($option_value !== '') {
danielebarchiesi@0 566 // Administer provided values.
danielebarchiesi@0 567 if (isset($options[$option_value])) {
danielebarchiesi@0 568 $items[] = $element['#format'] == 'html' ? _webform_filter_xss($options[$option_value]) : $options[$option_value];
danielebarchiesi@0 569 }
danielebarchiesi@0 570 // User-specified in the "other" field.
danielebarchiesi@0 571 else {
danielebarchiesi@0 572 $items[] = $element['#format'] == 'html' ? check_plain($option_value) : $option_value;
danielebarchiesi@0 573 }
danielebarchiesi@0 574 }
danielebarchiesi@0 575 }
danielebarchiesi@0 576 }
danielebarchiesi@0 577 else {
danielebarchiesi@0 578 if (isset($element['#value'][0]) && $element['#value'][0] !== '') {
danielebarchiesi@0 579 // Administer provided values.
danielebarchiesi@0 580 if (isset($options[$element['#value'][0]])) {
danielebarchiesi@0 581 $items[] = $element['#format'] == 'html' ? _webform_filter_xss($options[$element['#value'][0]]) : $options[$element['#value'][0]];
danielebarchiesi@0 582 }
danielebarchiesi@0 583 // User-specified in the "other" field.
danielebarchiesi@0 584 else {
danielebarchiesi@0 585 $items[] = $element['#format'] == 'html' ? check_plain($element['#value'][0]) : $element['#value'][0];
danielebarchiesi@0 586 }
danielebarchiesi@0 587 }
danielebarchiesi@0 588 }
danielebarchiesi@0 589
danielebarchiesi@0 590 if ($element['#format'] == 'html') {
danielebarchiesi@0 591 $output = count($items) > 1 ? theme('item_list', array('items' => $items)) : (isset($items[0]) ? $items[0] : ' ');
danielebarchiesi@0 592 }
danielebarchiesi@0 593 else {
danielebarchiesi@0 594 if (count($items) > 1) {
danielebarchiesi@0 595 foreach ($items as $key => $item) {
danielebarchiesi@0 596 $items[$key] = ' - ' . $item;
danielebarchiesi@0 597 }
danielebarchiesi@0 598 $output = implode("\n", $items);
danielebarchiesi@0 599 }
danielebarchiesi@0 600 else {
danielebarchiesi@0 601 $output = isset($items[0]) ? $items[0] : ' ';
danielebarchiesi@0 602 }
danielebarchiesi@0 603 }
danielebarchiesi@0 604
danielebarchiesi@0 605 return $output;
danielebarchiesi@0 606 }
danielebarchiesi@0 607
danielebarchiesi@0 608 /**
danielebarchiesi@0 609 * Implements _webform_analysis_component().
danielebarchiesi@0 610 */
danielebarchiesi@0 611 function _webform_analysis_select($component, $sids = array(), $single = FALSE) {
danielebarchiesi@0 612 $options = _webform_select_options($component, TRUE);
danielebarchiesi@0 613 $show_other_results = $single;
danielebarchiesi@0 614
danielebarchiesi@0 615 $sid_placeholders = count($sids) ? array_fill(0, count($sids), "'%s'") : array();
danielebarchiesi@0 616 $sid_filter = count($sids) ? " AND sid IN (" . implode(",", $sid_placeholders) . ")" : "";
danielebarchiesi@0 617
danielebarchiesi@0 618 $option_operator = $show_other_results ? 'NOT IN' : 'IN';
danielebarchiesi@0 619 $query = db_select('webform_submitted_data', 'wsd', array('fetch' => PDO::FETCH_ASSOC))
danielebarchiesi@0 620 ->fields('wsd', array('data'))
danielebarchiesi@0 621 ->condition('nid', $component['nid'])
danielebarchiesi@0 622 ->condition('cid', $component['cid'])
danielebarchiesi@0 623 ->condition('data', '', '<>')
danielebarchiesi@0 624 ->condition('data', array_keys($options), $option_operator)
danielebarchiesi@0 625 ->groupBy('data');
danielebarchiesi@0 626 $query->addExpression('COUNT(data)', 'datacount');
danielebarchiesi@0 627
danielebarchiesi@0 628 if (count($sids)) {
danielebarchiesi@0 629 $query->condition('sid', $sids, 'IN');
danielebarchiesi@0 630 }
danielebarchiesi@0 631
danielebarchiesi@0 632 $count_query = db_select('webform_submitted_data', 'wsd', array('fetch' => PDO::FETCH_ASSOC))
danielebarchiesi@0 633 ->condition('nid', $component['nid'])
danielebarchiesi@0 634 ->condition('cid', $component['cid'])
danielebarchiesi@0 635 ->condition('data', '', '<>');
danielebarchiesi@0 636 $count_query->addExpression('COUNT(*)', 'datacount');
danielebarchiesi@0 637 if (count($sids)) {
danielebarchiesi@0 638 $count_query->condition('sid', $sids, 'IN');
danielebarchiesi@0 639 }
danielebarchiesi@0 640
danielebarchiesi@0 641 $result = $query->execute();
danielebarchiesi@0 642 $rows = array();
danielebarchiesi@0 643 $normal_count = 0;
danielebarchiesi@0 644 foreach ($result as $data) {
danielebarchiesi@0 645 $display_option = $single ? $data['data'] : $options[$data['data']];
danielebarchiesi@0 646 $rows[$data['data']] = array(_webform_filter_xss($display_option), $data['datacount']);
danielebarchiesi@0 647 $normal_count += $data['datacount'];
danielebarchiesi@0 648 }
danielebarchiesi@0 649
danielebarchiesi@0 650 if (!$show_other_results) {
danielebarchiesi@0 651 // Order the results according to the normal options array.
danielebarchiesi@0 652 $ordered_rows = array();
danielebarchiesi@0 653 foreach (array_intersect_key($options, $rows) as $key => $label) {
danielebarchiesi@0 654 $ordered_rows[] = $rows[$key];
danielebarchiesi@0 655 }
danielebarchiesi@0 656
danielebarchiesi@0 657 // Add a row for any unknown or user-entered values.
danielebarchiesi@0 658 if ($component['extra']['other_option']) {
danielebarchiesi@0 659 $full_count = $count_query->execute()->fetchField();
danielebarchiesi@0 660 $other_count = $full_count - $normal_count;
danielebarchiesi@0 661 $display_option = !empty($component['extra']['other_text']) ? check_plain($component['extra']['other_text']) : t('Other...');
danielebarchiesi@0 662 $other_text = $other_count ? $other_count . ' (' . l(t('view'), 'node/' . $component['nid'] . '/webform-results/analysis/' . $component['cid']) . ')' : $other_count;
danielebarchiesi@0 663 $ordered_rows[] = array($display_option, $other_text);
danielebarchiesi@0 664 }
danielebarchiesi@0 665
danielebarchiesi@0 666 $rows = $ordered_rows;
danielebarchiesi@0 667 }
danielebarchiesi@0 668
danielebarchiesi@0 669 return $rows;
danielebarchiesi@0 670 }
danielebarchiesi@0 671
danielebarchiesi@0 672 /**
danielebarchiesi@0 673 * Implements _webform_table_component().
danielebarchiesi@0 674 */
danielebarchiesi@0 675 function _webform_table_select($component, $value) {
danielebarchiesi@0 676 // Convert submitted 'safe' values to un-edited, original form.
danielebarchiesi@0 677 $options = _webform_select_options($component, TRUE);
danielebarchiesi@0 678
danielebarchiesi@0 679 $value = (array) $value;
danielebarchiesi@0 680 $items = array();
danielebarchiesi@0 681 // Set the value as a single string.
danielebarchiesi@0 682 foreach ($value as $option_value) {
danielebarchiesi@0 683 if ($option_value !== '') {
danielebarchiesi@0 684 if (isset($options[$option_value])) {
danielebarchiesi@0 685 $items[] = _webform_filter_xss($options[$option_value]);
danielebarchiesi@0 686 }
danielebarchiesi@0 687 else {
danielebarchiesi@0 688 $items[] = check_plain($option_value);
danielebarchiesi@0 689 }
danielebarchiesi@0 690 }
danielebarchiesi@0 691 }
danielebarchiesi@0 692
danielebarchiesi@0 693 return implode('<br />', $items);
danielebarchiesi@0 694 }
danielebarchiesi@0 695
danielebarchiesi@0 696 /**
danielebarchiesi@0 697 * Implements _webform_csv_headers_component().
danielebarchiesi@0 698 */
danielebarchiesi@0 699 function _webform_csv_headers_select($component, $export_options) {
danielebarchiesi@0 700 $headers = array(
danielebarchiesi@0 701 0 => array(),
danielebarchiesi@0 702 1 => array(),
danielebarchiesi@0 703 2 => array(),
danielebarchiesi@0 704 );
danielebarchiesi@0 705
danielebarchiesi@0 706 if ($component['extra']['multiple'] && $export_options['select_format'] == 'separate') {
danielebarchiesi@0 707 $headers[0][] = '';
danielebarchiesi@0 708 $headers[1][] = $component['name'];
danielebarchiesi@0 709 $items = _webform_select_options($component, TRUE, FALSE);
danielebarchiesi@0 710 if ($component['extra']['other_option']) {
danielebarchiesi@0 711 $other_label = !empty($component['extra']['other_text']) ? check_plain($component['extra']['other_text']) : t('Other...');
danielebarchiesi@0 712 $items[$other_label] = $other_label;
danielebarchiesi@0 713 }
danielebarchiesi@0 714 $count = 0;
danielebarchiesi@0 715 foreach ($items as $key => $item) {
danielebarchiesi@0 716 // Empty column per sub-field in main header.
danielebarchiesi@0 717 if ($count != 0) {
danielebarchiesi@0 718 $headers[0][] = '';
danielebarchiesi@0 719 $headers[1][] = '';
danielebarchiesi@0 720 }
danielebarchiesi@0 721 if ($export_options['select_keys']) {
danielebarchiesi@0 722 $headers[2][] = $key;
danielebarchiesi@0 723 }
danielebarchiesi@0 724 else {
danielebarchiesi@0 725 $headers[2][] = $item;
danielebarchiesi@0 726 }
danielebarchiesi@0 727 $count++;
danielebarchiesi@0 728 }
danielebarchiesi@0 729 }
danielebarchiesi@0 730 else {
danielebarchiesi@0 731 $headers[0][] = '';
danielebarchiesi@0 732 $headers[1][] = '';
danielebarchiesi@0 733 $headers[2][] = $component['name'];
danielebarchiesi@0 734 }
danielebarchiesi@0 735 return $headers;
danielebarchiesi@0 736 }
danielebarchiesi@0 737
danielebarchiesi@0 738 /**
danielebarchiesi@0 739 * Implements _webform_csv_data_component().
danielebarchiesi@0 740 */
danielebarchiesi@0 741 function _webform_csv_data_select($component, $export_options, $value) {
danielebarchiesi@0 742 $options = _webform_select_options($component, TRUE, FALSE);
danielebarchiesi@0 743 $return = array();
danielebarchiesi@0 744
danielebarchiesi@0 745 if ($component['extra']['multiple']) {
danielebarchiesi@0 746 foreach ($options as $key => $item) {
danielebarchiesi@0 747 $index = array_search($key, (array) $value);
danielebarchiesi@0 748 if ($index !== FALSE) {
danielebarchiesi@0 749 if ($export_options['select_format'] == 'separate') {
danielebarchiesi@0 750 $return[] = 'X';
danielebarchiesi@0 751 }
danielebarchiesi@0 752 else {
danielebarchiesi@0 753 $return[] = $export_options['select_keys'] ? $key : $item;
danielebarchiesi@0 754 }
danielebarchiesi@0 755 unset($value[$index]);
danielebarchiesi@0 756 }
danielebarchiesi@0 757 elseif ($export_options['select_format'] == 'separate') {
danielebarchiesi@0 758 $return[] = '';
danielebarchiesi@0 759 }
danielebarchiesi@0 760 }
danielebarchiesi@0 761
danielebarchiesi@0 762 // Any remaining items in the $value array will be user-added options.
danielebarchiesi@0 763 if ($component['extra']['other_option']) {
danielebarchiesi@0 764 $return[] = count($value) ? implode(',', $value) : '';
danielebarchiesi@0 765 }
danielebarchiesi@0 766 }
danielebarchiesi@0 767 else {
danielebarchiesi@0 768 $key = $value[0];
danielebarchiesi@0 769 if ($export_options['select_keys']) {
danielebarchiesi@0 770 $return = $key;
danielebarchiesi@0 771 }
danielebarchiesi@0 772 else {
danielebarchiesi@0 773 $return = isset($options[$key]) ? $options[$key] : $key;
danielebarchiesi@0 774 }
danielebarchiesi@0 775 }
danielebarchiesi@0 776
danielebarchiesi@0 777 if ($component['extra']['multiple'] && $export_options['select_format'] == 'compact') {
danielebarchiesi@0 778 $return = implode(',', (array) $return);
danielebarchiesi@0 779 }
danielebarchiesi@0 780
danielebarchiesi@0 781 return $return;
danielebarchiesi@0 782 }
danielebarchiesi@0 783
danielebarchiesi@0 784 /**
danielebarchiesi@0 785 * Menu callback; Return a predefined list of select options as JSON.
danielebarchiesi@0 786 */
danielebarchiesi@0 787 function webform_select_options_ajax($source_name = '') {
danielebarchiesi@0 788 $info = _webform_select_options_info();
danielebarchiesi@0 789
danielebarchiesi@0 790 $component['extra']['options_source'] = $source_name;
danielebarchiesi@0 791 if ($source_name && isset($info[$source_name])) {
danielebarchiesi@0 792 $options = _webform_select_options_to_text(_webform_select_options($component, !$component['extra']['aslist'], FALSE));
danielebarchiesi@0 793 }
danielebarchiesi@0 794 else {
danielebarchiesi@0 795 $options = '';
danielebarchiesi@0 796 }
danielebarchiesi@0 797
danielebarchiesi@0 798 $return = array(
danielebarchiesi@0 799 'elementId' => module_exists('options_element') ? 'edit-items-options-options-field-widget' : 'edit-extra-items',
danielebarchiesi@0 800 'options' => $options,
danielebarchiesi@0 801 );
danielebarchiesi@0 802
danielebarchiesi@0 803 drupal_json_output($return);
danielebarchiesi@0 804 }
danielebarchiesi@0 805
danielebarchiesi@0 806 /**
danielebarchiesi@0 807 * Generate a list of options for a select list.
danielebarchiesi@0 808 */
danielebarchiesi@0 809 function _webform_select_options($component, $flat = FALSE, $filter = TRUE) {
danielebarchiesi@0 810 if ($component['extra']['options_source']) {
danielebarchiesi@0 811 $options = _webform_select_options_callback($component['extra']['options_source'], $component, $flat, $filter);
danielebarchiesi@0 812 }
danielebarchiesi@0 813 else {
danielebarchiesi@0 814 $options = _webform_select_options_from_text($component['extra']['items'], $flat, $filter);
danielebarchiesi@0 815 }
danielebarchiesi@0 816
danielebarchiesi@0 817 return isset($options) ? $options : array();
danielebarchiesi@0 818 }
danielebarchiesi@0 819
danielebarchiesi@0 820 /**
danielebarchiesi@0 821 * Load Webform select option info from 3rd party modules.
danielebarchiesi@0 822 */
danielebarchiesi@0 823 function _webform_select_options_info() {
danielebarchiesi@0 824 static $info;
danielebarchiesi@0 825 if (!isset($info)) {
danielebarchiesi@0 826 $info = array();
danielebarchiesi@0 827
danielebarchiesi@0 828 foreach (module_implements('webform_select_options_info') as $module) {
danielebarchiesi@0 829 $additions = module_invoke($module, 'webform_select_options_info');
danielebarchiesi@0 830 foreach ($additions as $key => $addition) {
danielebarchiesi@0 831 $additions[$key]['module'] = $module;
danielebarchiesi@0 832 }
danielebarchiesi@0 833 $info = array_merge($info, $additions);
danielebarchiesi@0 834 }
danielebarchiesi@0 835 drupal_alter('webform_select_options_info', $info);
danielebarchiesi@0 836 }
danielebarchiesi@0 837 return $info;
danielebarchiesi@0 838 }
danielebarchiesi@0 839
danielebarchiesi@0 840 /**
danielebarchiesi@0 841 * Execute a select option callback.
danielebarchiesi@0 842 *
danielebarchiesi@0 843 * @param $name
danielebarchiesi@0 844 * The name of the options group.
danielebarchiesi@0 845 * @param $component
danielebarchiesi@0 846 * The full Webform component.
danielebarchiesi@0 847 * @param $flat
danielebarchiesi@0 848 * Whether the information returned should exclude any nested groups.
danielebarchiesi@0 849 * @param $filter
danielebarchiesi@0 850 * Whether information returned should be sanitized. Defaults to TRUE.
danielebarchiesi@0 851 */
danielebarchiesi@0 852 function _webform_select_options_callback($name, $component, $flat = FALSE, $filter = TRUE) {
danielebarchiesi@0 853 $info = _webform_select_options_info();
danielebarchiesi@0 854
danielebarchiesi@0 855 // Include any necessary files.
danielebarchiesi@0 856 if (isset($info[$name]['file'])) {
danielebarchiesi@0 857 $pathinfo = pathinfo($info[$name]['file']);
danielebarchiesi@0 858 $path = ($pathinfo['dirname'] ? $pathinfo['dirname'] . '/' : '') . basename($pathinfo['basename'], '.' . $pathinfo['extension']);
danielebarchiesi@0 859 module_load_include($pathinfo['extension'], $info[$name]['module'], $path);
danielebarchiesi@0 860 }
danielebarchiesi@0 861
danielebarchiesi@0 862 // Execute the callback function.
danielebarchiesi@0 863 if (isset($info[$name]['options callback']) && function_exists($info[$name]['options callback'])) {
danielebarchiesi@0 864 $function = $info[$name]['options callback'];
danielebarchiesi@0 865
danielebarchiesi@0 866 $arguments = array();
danielebarchiesi@0 867 if (isset($info[$name]['options arguments'])) {
danielebarchiesi@0 868 $arguments = $info[$name]['options arguments'];
danielebarchiesi@0 869 }
danielebarchiesi@0 870
danielebarchiesi@0 871 return $function($component, $flat, $filter, $arguments);
danielebarchiesi@0 872 }
danielebarchiesi@0 873 }
danielebarchiesi@0 874
danielebarchiesi@0 875 /**
danielebarchiesi@0 876 * Utility function to split user-entered values from new-line separated
danielebarchiesi@0 877 * text into an array of options.
danielebarchiesi@0 878 *
danielebarchiesi@0 879 * @param $text
danielebarchiesi@0 880 * Text to be converted into a select option array.
danielebarchiesi@0 881 * @param $flat
danielebarchiesi@0 882 * Optional. If specified, return the option array and exclude any optgroups.
danielebarchiesi@0 883 * @param $filter
danielebarchiesi@0 884 * Optional. Whether or not to filter returned values.
danielebarchiesi@0 885 */
danielebarchiesi@0 886 function _webform_select_options_from_text($text, $flat = FALSE, $filter = TRUE) {
danielebarchiesi@0 887 static $option_cache = array();
danielebarchiesi@0 888
danielebarchiesi@0 889 // Keep each processed option block in an array indexed by the MD5 hash of
danielebarchiesi@0 890 // the option text and the value of the $flat variable.
danielebarchiesi@0 891 $md5 = md5($text);
danielebarchiesi@0 892
danielebarchiesi@0 893 // Check if this option block has been previously processed.
danielebarchiesi@0 894 if (!isset($option_cache[$flat][$md5])) {
danielebarchiesi@0 895 $options = array();
danielebarchiesi@0 896 $rows = array_filter(explode("\n", trim($text)));
danielebarchiesi@0 897 $group = NULL;
danielebarchiesi@0 898 foreach ($rows as $option) {
danielebarchiesi@0 899 $option = trim($option);
danielebarchiesi@0 900 /**
danielebarchiesi@0 901 * If the Key of the option is within < >, treat as an optgroup
danielebarchiesi@0 902 *
danielebarchiesi@0 903 * <Group 1>
danielebarchiesi@0 904 * creates an optgroup with the label "Group 1"
danielebarchiesi@0 905 *
danielebarchiesi@0 906 * <>
danielebarchiesi@0 907 * Unsets the current group, allowing items to be inserted at the root element.
danielebarchiesi@0 908 */
danielebarchiesi@0 909 if (preg_match('/^\<([^>]*)\>$/', $option, $matches)) {
danielebarchiesi@0 910 if (empty($matches[1])) {
danielebarchiesi@0 911 unset($group);
danielebarchiesi@0 912 }
danielebarchiesi@0 913 elseif (!$flat) {
danielebarchiesi@0 914 $group = $filter ? _webform_filter_values($matches[1], NULL, NULL, NULL, FALSE) : $matches[1];
danielebarchiesi@0 915 }
danielebarchiesi@0 916 }
danielebarchiesi@0 917 elseif (preg_match('/^([^|]+)\|(.*)$/', $option, $matches)) {
danielebarchiesi@0 918 $key = $filter ? _webform_filter_values($matches[1], NULL, NULL, NULL, FALSE) : $matches[1];
danielebarchiesi@0 919 $value = $filter ? _webform_filter_values($matches[2], NULL, NULL, NULL, FALSE) : $matches[2];
danielebarchiesi@0 920 isset($group) ? $options[$group][$key] = $value : $options[$key] = $value;
danielebarchiesi@0 921 }
danielebarchiesi@0 922 else {
danielebarchiesi@0 923 $filtered_option = $filter ? _webform_filter_values($option, NULL, NULL, NULL, FALSE) : $option;
danielebarchiesi@0 924 isset($group) ? $options[$group][$filtered_option] = $filtered_option : $options[$filtered_option] = $filtered_option;
danielebarchiesi@0 925 }
danielebarchiesi@0 926 }
danielebarchiesi@0 927
danielebarchiesi@0 928 $option_cache[$flat][$md5] = $options;
danielebarchiesi@0 929 }
danielebarchiesi@0 930
danielebarchiesi@0 931 // Return our options from the option_cache array.
danielebarchiesi@0 932 return $option_cache[$flat][$md5];
danielebarchiesi@0 933 }
danielebarchiesi@0 934
danielebarchiesi@0 935 /**
danielebarchiesi@0 936 * Convert an array of options into text.
danielebarchiesi@0 937 */
danielebarchiesi@0 938 function _webform_select_options_to_text($options) {
danielebarchiesi@0 939 $output = '';
danielebarchiesi@0 940 $previous_key = FALSE;
danielebarchiesi@0 941
danielebarchiesi@0 942 foreach ($options as $key => $value) {
danielebarchiesi@0 943 // Convert groups.
danielebarchiesi@0 944 if (is_array($value)) {
danielebarchiesi@0 945 $output .= '<' . $key . '>' . "\n";
danielebarchiesi@0 946 foreach ($value as $subkey => $subvalue) {
danielebarchiesi@0 947 $output .= $subkey . '|' . $subvalue . "\n";
danielebarchiesi@0 948 }
danielebarchiesi@0 949 $previous_key = $key;
danielebarchiesi@0 950 }
danielebarchiesi@0 951 // Typical key|value pairs.
danielebarchiesi@0 952 else {
danielebarchiesi@0 953 // Exit out of any groups.
danielebarchiesi@0 954 if (isset($options[$previous_key]) && is_array($options[$previous_key])) {
danielebarchiesi@0 955 $output .= "<>\n";
danielebarchiesi@0 956 }
danielebarchiesi@0 957 // Skip empty rows.
danielebarchiesi@0 958 if ($options[$key] !== '') {
danielebarchiesi@0 959 $output .= $key . '|' . $value . "\n";
danielebarchiesi@0 960 }
danielebarchiesi@0 961 $previous_key = $key;
danielebarchiesi@0 962 }
danielebarchiesi@0 963 }
danielebarchiesi@0 964
danielebarchiesi@0 965 return $output;
danielebarchiesi@0 966 }
danielebarchiesi@0 967
danielebarchiesi@0 968 /**
danielebarchiesi@0 969 * Utility function to shuffle an array while preserving key-value pairs.
danielebarchiesi@0 970 */
danielebarchiesi@0 971 function _webform_shuffle_options(&$array) {
danielebarchiesi@0 972 // First shuffle the array keys, then use them as the basis for ordering
danielebarchiesi@0 973 // the options.
danielebarchiesi@0 974 $aux = array();
danielebarchiesi@0 975 $keys = array_keys($array);
danielebarchiesi@0 976 shuffle($keys);
danielebarchiesi@0 977 foreach ($keys as $key) {
danielebarchiesi@0 978 $aux[$key] = $array[$key];
danielebarchiesi@0 979 }
danielebarchiesi@0 980 $array = $aux;
danielebarchiesi@0 981 }