Projet

Général

Profil

Paste
Télécharger (16,5 ko) Statistiques
| Branche: | Révision:

root / drupal7 / sites / all / modules / webform_validation / webform_validation.module @ a2bb1a14

1
<?php
2

    
3
/**
4
 * @file
5
 * Add validation rules to webforms
6
 */
7

    
8
include_once('webform_validation.validators.inc');
9
include_once('webform_validation.rules.inc');
10

    
11
/**
12
 * Implements hook_menu().
13
 */
14
function webform_validation_menu() {
15
  $items = array();
16

    
17
  $items['node/%webform_menu/webform/validation'] = array(
18
    'title' => 'Form validation',
19
    'page callback' => 'webform_validation_manage',
20
    'page arguments' => array(1),
21
    'access callback' => 'node_access',
22
    'access arguments' => array('update', 1),
23
    'file' => 'webform_validation.admin.inc',
24
    'weight' => 3,
25
    'type' => MENU_LOCAL_TASK,
26
  );
27

    
28
  $items['node/%webform_menu/webform/validation/add/%'] = array(
29
    'title' => 'Add validation',
30
    'page callback' => 'drupal_get_form',
31
    'page arguments' => array('webform_validation_manage_rule', 1, 'add', 5),
32
    'access callback' => 'node_access',
33
    'access arguments' => array('update', 1),
34
    'file' => 'webform_validation.admin.inc',
35
    'type' => MENU_CALLBACK,
36
  );
37

    
38
  $items['node/%webform_menu/webform/validation/edit/%/%webform_validation_rule'] = array(
39
    'title' => 'Edit rule',
40
    'page callback' => 'drupal_get_form',
41
    'page arguments' => array('webform_validation_manage_rule', 1, 'edit', 5, 6),
42
    'access callback' => 'node_access',
43
    'access arguments' => array('update', 1),
44
    'file' => 'webform_validation.admin.inc',
45
    'type' => MENU_CALLBACK,
46
  );
47

    
48
  $items['node/%webform_menu/webform/validation/delete/%webform_validation_rule'] = array(
49
    'title' => 'Delete rule',
50
    'page callback' => 'drupal_get_form',
51
    'page arguments' => array('webform_validation_delete_rule', 5),
52
    'access callback' => 'node_access',
53
    'access arguments' => array('update', 1),
54
    'file' => 'webform_validation.admin.inc',
55
    'type' => MENU_CALLBACK,
56
  );
57

    
58
  return $items;
59
}
60

    
61
/**
62
 * Loads validation rule from menu parameter
63
 */
64
function webform_validation_rule_load($ruleid) {
65
  return webform_validation_get_rule($ruleid);
66
}
67

    
68

    
69
/**
70
 * Implements hook_theme().
71
 */
72
function webform_validation_theme() {
73
  return array(
74
    'webform_validation_manage_add_rule' => array(
75
      'variables' => array(
76
        'nid' => NULL,
77
      ),
78
    ),
79
    'webform_validation_manage_overview_form' => array(
80
      'render element' => 'form',
81
    ),
82
  );
83
}
84

    
85
/**
86
 * Implements hook_form_alter().
87
 */
88
function webform_validation_form_alter(&$form, &$form_state, $form_id) {
89
  if (strpos($form_id, 'webform_client_form_') !== FALSE) {
90
    $form['#validate'][] = 'webform_validation_validate';
91
  }
92
}
93

    
94
/**
95
 * Implements hook_i18n_string_info().
96
 */
97
function webform_validation_i18n_string_info() {
98
  $groups = array();
99
  $groups['webform_validation'] = array(
100
    'title' => t('Webform Validation'),
101
    'description' => t('Translatable strings for webform validation translation'),
102
    'format' => FALSE, // This group doesn't have strings with format
103
    'list' => FALSE, // This group cannot list all strings
104
    'refresh callback' => 'webform_validation_i18n_string_refresh',
105
  );
106
  return $groups;
107
}
108

    
109
/**
110
 * Webform validation handler to validate against the given rules
111
 */
112
function webform_validation_validate($form, &$form_state) {
113
  $static_error_messages = &drupal_static(__FUNCTION__, array());
114
  $page_count = 1;
115
  $nid = $form_state['values']['details']['nid'];
116
  $node = node_load($nid);
117
  $values = isset($form_state['values']['submitted']) ? $form_state['values']['submitted'] : NULL;
118
  $flat_values = _webform_client_form_submit_flatten($node, $values);
119
  $rules = webform_validation_get_node_rules($nid);
120
  $sid = empty($form_state['values']['details']['sid']) ? 0 : $form_state['values']['details']['sid'];
121

    
122
  // Get number of pages for this webform
123
  if (isset($form_state['webform']['page_count'])) {
124
    $page_count = $form_state['webform']['page_count'];
125
  }
126
  elseif (isset($form_state['storage']['page_count'])) {
127
    $page_count = $form_state['storage']['page_count'];
128
  }
129

    
130
  // Filter out rules that don't apply to this step in the multistep form
131
  if ($values && $page_count && $page_count > 1) {
132
    $current_page_components = webform_validation_get_field_keys($form_state['values']['submitted'], $node);
133
    if ($rules) {
134
       // filter out rules that don't belong in the current step
135
      foreach ($rules as $ruleid => $rule) {
136
        // get all the component formkeys for this specific validation rule
137
        $rule_formkeys = webform_validation_rule_get_formkeys($rule);
138
        $rule_applies_to_current_page = FALSE;
139
        if (!empty($rule_formkeys)) {
140
          foreach ($rule_formkeys as $formkey) {
141
            if (in_array($formkey, $current_page_components)) {
142
              // this rule applies to the current page,
143
              // because one of the rule components is on the page
144
              $rule_applies_to_current_page = TRUE;
145
            }
146
          }
147
        }
148

    
149
        if (!$rule_applies_to_current_page) {
150
          unset($rules[$ruleid]);
151
        }
152
      }
153
    }
154
  }
155

    
156
  if ($rules) {
157
    $component_definitions = webform_validation_prefix_keys($node->webform['components']);
158

    
159
    // Remove hidden components.
160
    if (defined('WebformConditionals::componentShown')) {
161
      // New conditionals system.
162
      $sorter = webform_get_conditional_sorter($node);
163
      // If the form was retrieved from the form cache, the conditionals may not
164
      // have been executed yet.
165
      if (!$sorter->isExecuted()) {
166
        $sorter->executeConditionals(array(), 0);
167
      }
168
      foreach ($component_definitions as $key => $component) {
169
        if ($sorter->componentVisibility($component['cid'], $component['page_num']) !== WebformConditionals::componentShown) {
170
          unset($flat_values[$key]);
171
        }
172
      }
173
    }
174
    else {
175
      // Old conditionals system removed in Webform 7.x-4.8.
176
      // Webform 7.x-3.x does not define WEBFORM_CONDITIONAL_INCLUDE. Define if needed.
177
      if (!defined('WEBFORM_CONDITIONAL_INCLUDE')) {
178
        define('WEBFORM_CONDITIONAL_INCLUDE', 1);
179
      }
180
      foreach ($component_definitions as $key => $component) {
181
        // In Webform 7.x-3.x, _webform_client_form_rule_check() returns boolean.
182
        // Cast to int so that the function behaves as it does in 7.x-4.x.
183
        if (isset($flat_values[$key]) && (int) _webform_client_form_rule_check($node, $component, 0, $form_state['values']['submitted']) !== WEBFORM_CONDITIONAL_INCLUDE) {
184
          unset($flat_values[$key]);
185
        }
186
      }
187
    }
188

    
189
    foreach ($rules as $rule) {
190
      // create a list of components that need validation against this rule (component id => user submitted value)
191
      $items = array();
192
      foreach ($rule['components'] as $cid => $component) {
193
        if (array_key_exists($cid, $flat_values)) {
194
          $items[$cid] = $flat_values[$cid];
195
        }
196
      }
197
      // prefix array keys to avoid reindexing by the module_invoke_all function call
198
      $items = webform_validation_prefix_keys($items);
199
      $rule['sid'] = $sid;
200
      // have the submitted values validated
201
      $errors = module_invoke_all("webform_validation_validate", $rule['validator'], $items, $component_definitions, $rule);
202
      if ($errors) {
203
        $errors = webform_validation_unprefix_keys($errors);
204
        $components = webform_validation_unprefix_keys($component_definitions);
205
        foreach ($errors as $item_key => $error) {
206
          // Do not set error message if an identical message has already been set.
207
          if (in_array($error, $static_error_messages, TRUE)) {
208
            continue;
209
          }
210
          $static_error_messages[] = $error;
211

    
212
          // build the proper form element error key, taking into account hierarchy
213
          $error_key = 'submitted][' . webform_validation_parent_tree($item_key, $components) . $components[$item_key]['form_key'];
214
          if (is_array($error)) {
215
            foreach ($error as $sub_item_key => $sub_error) {
216
              form_set_error($error_key . '][' . $sub_item_key, $sub_error);
217
            }
218
          }
219
          else {
220
            // @ignore security_form_set_error. filter_xss() is run in _webform_validation_i18n_error_message().
221
            form_set_error($error_key, $error);
222
          }
223
        }
224
      }
225
    }
226
  }
227
}
228

    
229
/**
230
 * Recursive helper function to get all field keys (including fields in fieldsets)
231
 */
232
function webform_validation_get_field_keys($submitted, $node) {
233
  static $fields = array();
234
  foreach (element_children($submitted) as $child) {
235
    if (is_array($submitted[$child]) && element_children($submitted[$child])) {
236
      // only keep searching recursively if it's a fieldset
237
      $group_components = _webform_validation_get_group_types();
238
      if (in_array(_webform_validation_get_component_type($node, $child), $group_components)) {
239
        webform_validation_get_field_keys($submitted[$child], $node);
240
      }
241
      else {
242
        $fields[$child] = $child;
243
      }
244

    
245
    }
246
    else {
247
      $fields[$child] = $child;
248
    }
249
  }
250
  return $fields;
251
}
252

    
253
/**
254
 * Recursively add the parents for the element, to be used as first argument to form_set_error
255
 */
256
function webform_validation_parent_tree($cid, $components) {
257
  $output = '';
258
  if ($pid = $components[$cid]['pid']) {
259
    $output .= webform_validation_parent_tree($pid, $components);
260
    $output .= $components[$pid]['form_key'] . '][';
261
  }
262
  return $output;
263
}
264

    
265
/**
266
 * Get an array of formkeys for all components that have been assigned to a rule
267
 */
268
function webform_validation_rule_get_formkeys($rule) {
269
  $formkeys = array();
270
  if (isset($rule['components'])) {
271
    foreach ($rule['components'] as $cid => $component) {
272
      $formkeys[] = $component['form_key'];
273
    }
274
  }
275
  return $formkeys;
276
}
277

    
278
/**
279
 * Prefix numeric array keys to avoid them being reindexed by module_invoke_all
280
 */
281
function webform_validation_prefix_keys($arr) {
282
  $ret = array();
283
  foreach ($arr as $k => $v) {
284
    $ret['item_' . $k] = $v;
285
  }
286
  return $ret;
287
}
288

    
289
/**
290
 * Undo prefixing numeric array keys to avoid them being reindexed by module_invoke_all
291
 */
292
function webform_validation_unprefix_keys($arr) {
293
  $ret = array();
294
  foreach ($arr as $k => $v) {
295
    $new_key = str_replace('item_', '', $k);
296
    $ret[$new_key] = $v;
297
  }
298
  return $ret;
299
}
300
/**
301
 * Theme the 'add rule' list
302
 */
303
function theme_webform_validation_manage_add_rule($variables) {
304
  $nid = $variables['nid'];
305
  $output = '';
306
  $validators = webform_validation_get_validators();
307

    
308
  if ($validators) {
309
    $results = db_query('SELECT DISTINCT type FROM {webform_component} WHERE nid = :nid', array('nid' => $nid));
310
    $types = array();
311
    while ($item = $results->fetch()) {
312
      $types[] = $item->type;
313
    }
314

    
315
    $output = '<h3>' . t('Add a validation rule') . '</h3>';
316
    $output .= '<dl>';
317
    foreach ($validators as $validator_key => $validator_info) {
318
      $url = 'node/' . $nid . '/webform/validation/add/' . $validator_key;
319
      if (array_intersect($types, $validator_info['component_types'])) {
320
        $title = l($validator_info['name'], $url, array('query' => drupal_get_destination()));
321
        $component_list_postfix = '';
322
      }
323
      else {
324
        $title = $validator_info['name'];
325
        $component_list_postfix = '; ' . t('none present in this form');
326
      }
327
      $item = '<dt>' . $title . '</dt>';
328
      $item .= '<dd>';
329
      $item .= $validator_info['description'];
330
      $item .= ' ' . t('Works with: @component_types.', array('@component_types' => implode(', ', $validator_info['component_types']) . $component_list_postfix)) . '</dd>';
331
      $output .= $item;
332
    }
333
    $output .= '</dl>';
334
  }
335
  return $output;
336
}
337

    
338
/**
339
 * Implements hook_webform_validation().
340
 */
341
function webform_validation_webform_validation($type, $op, $data) {
342
  if ($type == 'rule' && in_array($op, array('add', 'edit'))) {
343
    if (module_exists('i18n_string') && isset($data['error_message'])) {
344
      i18n_string_update('webform_validation:error_message:' . $data['ruleid'] . ':message', $data['error_message']);
345
    }
346
  }
347
}
348

    
349
/**
350
 * Implements hook_node_insert().
351
 */
352
function webform_validation_node_insert($node) {
353
  if (module_exists('clone') && in_array($node->type, webform_variable_get('webform_node_types'))) {
354
    webform_validation_node_clone($node);
355
  }
356
}
357

    
358
/**
359
 * Implements hook_node_delete().
360
 */
361
function webform_validation_node_delete($node) {
362
  $rules = webform_validation_get_node_rules($node->nid);
363
  if ($rules) {
364
    foreach (array_keys($rules) as $ruleid) {
365
      webform_dynamic_delete_rule($ruleid);
366
    }
367
  }
368
}
369

    
370
/**
371
 * Adds support for node_clone module
372
 */
373
function webform_validation_node_clone($node) {
374
  if (!in_array($node->type, webform_variable_get('webform_node_types'))) {
375
    return;
376
  }
377
  if (isset($node->clone_from_original_nid)) {
378
    $original_nid = $node->clone_from_original_nid;
379
    // Get existing rules for original node
380
    $rules = webform_validation_get_node_rules($original_nid);
381
    if ($rules) {
382
      foreach ($rules as $orig_ruleid => $rule) {
383
        unset($rule['ruleid']);
384
        $rule['action'] = 'add';
385
        $rule['nid'] = $node->nid; // attach existing rules to new node
386
        $rule['rule_components'] = $rule['components'];
387
        webform_validation_rule_save($rule);
388
      }
389
    }
390
  }
391
}
392

    
393
/**
394
 * Save a validation rule. Data comes from the admin form or nodeapi function in
395
 * case of node clone.
396
 *
397
 * @param array $values
398
 *   An associative array containing:
399
 *   - action: "add" or "edit".
400
 *   - ruleid: ID of the rule to edit. Do not set for "add".
401
 *   - nid: Node ID of the Webform.
402
 *   - validator: Machine name of the validator used by this validation rule.
403
 *   - rulename: Human-readable name for this validation rule.
404
 *   - rule_components: An array in which the keys and the values are the cid's
405
 *     of the Webform components that this rule applies to.
406
 *
407
 * @return int
408
 *   The $ruleid of the rule added or edited.
409
 */
410
function webform_validation_rule_save($values) {
411
  if ($values['action'] === 'add') {
412
    $primary_keys = array();
413
  }
414
  elseif ($values['action'] === 'edit') {
415
    $primary_keys = array('ruleid');
416
  }
417
  else {
418
    return FALSE;
419
  }
420

    
421
  drupal_write_record('webform_validation_rule', $values, $primary_keys);
422

    
423
  // Delete existing component records for this ruleid.
424
  if ($values['action'] === 'edit') {
425
    db_delete('webform_validation_rule_components')
426
      ->condition('ruleid', $values['ruleid'])
427
      ->execute();
428
  }
429

    
430
  $components = array_filter($values['rule_components']);
431
  if ($values['ruleid'] && $components) {
432
    webform_validation_save_rule_components($values['ruleid'], $components);
433
    module_invoke_all('webform_validation', 'rule', $values['action'], $values);
434
  }
435

    
436
  return $values['ruleid'];
437
}
438

    
439
/**
440
 * Save components attached to a specific rule.
441
 *
442
 * @param int $ruleid
443
 *   The ruleid of the rule being saved.
444
 * @param array $components
445
 *   An array in which the keys are the cid's of the components attached to the rule.
446
 * @return array
447
 *   An array of the return statuses for each query keyed by cid.
448
 */
449
function webform_validation_save_rule_components($ruleid, $components) {
450
  $return_status = array();
451
  foreach ($components as $cid => $component) {
452
    $return_status[$cid] = db_merge('webform_validation_rule_components')
453
      ->key(array(
454
        'ruleid' => $ruleid,
455
        'cid' => $cid,
456
      ))
457
      ->fields(array(
458
        'ruleid' => $ruleid,
459
        'cid' => $cid,
460
      ))
461
      ->execute();
462
  }
463
  return $return_status;
464
}
465

    
466
/**
467
 * Given a webform node, get the component type based on a given component key
468
 */
469
function _webform_validation_get_component_type($node, $component_key) {
470
  if ($node->webform['components']) {
471
    foreach ($node->webform['components'] as $component) {
472
      if ($component['form_key'] == $component_key) {
473
        return $component['type'];
474
      }
475
    }
476
  }
477
  return FALSE;
478
}
479

    
480
/**
481
 * Get all webform components that are defined as a group
482
 */
483
function _webform_validation_get_group_types() {
484
  $types = array();
485
  foreach (webform_components() as $name => $component) {
486
    if (isset($component['features']['group']) && $component['features']['group']) {
487
      $types[] = $name;
488
    }
489
  }
490
  return $types;
491
}
492

    
493
/**
494
 * Implements hook_webform_validator_alter().
495
 */
496
function webform_validation_webform_validator_alter(&$validators) {
497
  // Add support for the Select (or Other) module
498
  if (module_exists('select_or_other')) {
499
    // if this module exists, all select components can now except user input.
500
    // Thus we provide those components the same rules as a textfield
501
    if ($validators) {
502
      foreach ($validators as $validator_name => $validator_info) {
503
        if (in_array('textfield', $validator_info['component_types'])) {
504
          $validators[$validator_name]['component_types'][] = 'select';
505
        }
506
        $validators[$validator_name]['component_types'] = array_unique($validators[$validator_name]['component_types']);
507
      }
508
    }
509
  }
510
}