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 <Group Name>. <> 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 }
|