annotate sites/all/modules/webform/includes/webform.report.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 * This file includes helper functions for creating reports for webform.module
danielebarchiesi@0 6 *
danielebarchiesi@0 7 * @author Nathan Haug <nate@lullabot.com>
danielebarchiesi@0 8 */
danielebarchiesi@0 9
danielebarchiesi@0 10 // All functions within this file need the webform.submissions.inc.
danielebarchiesi@0 11 module_load_include('inc', 'webform', 'includes/webform.submissions');
danielebarchiesi@0 12
danielebarchiesi@0 13 /**
danielebarchiesi@0 14 * Retrieve lists of submissions for a given webform.
danielebarchiesi@0 15 */
danielebarchiesi@0 16 function webform_results_submissions($node, $user_filter, $pager_count) {
danielebarchiesi@0 17 global $user;
danielebarchiesi@0 18
danielebarchiesi@0 19 if (isset($_GET['results']) && is_numeric($_GET['results'])) {
danielebarchiesi@0 20 $pager_count = $_GET['results'];
danielebarchiesi@0 21 }
danielebarchiesi@0 22
danielebarchiesi@0 23 $header = theme('webform_results_submissions_header', array('node' => $node));
danielebarchiesi@0 24 if ($user_filter) {
danielebarchiesi@0 25 if ($user->uid) {
danielebarchiesi@0 26 drupal_set_title(t('Submissions for %user', array('%user' => $user->name)), PASS_THROUGH);
danielebarchiesi@0 27 }
danielebarchiesi@0 28 else {
danielebarchiesi@0 29 drupal_set_title(t('Your submissions'));
danielebarchiesi@0 30 webform_disable_page_cache();
danielebarchiesi@0 31 }
danielebarchiesi@0 32 webform_set_breadcrumb($node);
danielebarchiesi@0 33 $submissions = webform_get_submissions(array('nid' => $node->nid, 'uid' => $user->uid), $header, $pager_count);
danielebarchiesi@0 34 $count = webform_get_submission_count($node->nid, $user->uid);
danielebarchiesi@0 35 }
danielebarchiesi@0 36 else {
danielebarchiesi@0 37 $submissions = webform_get_submissions($node->nid, $header, $pager_count);
danielebarchiesi@0 38 $count = webform_get_submission_count($node->nid);
danielebarchiesi@0 39 }
danielebarchiesi@0 40
danielebarchiesi@0 41 $operation_column = end($header);
danielebarchiesi@0 42 $operation_total = $operation_column['colspan'];
danielebarchiesi@0 43
danielebarchiesi@0 44 $rows = array();
danielebarchiesi@0 45 foreach ($submissions as $sid => $submission) {
danielebarchiesi@0 46 $row = array(
danielebarchiesi@0 47 $submission->is_draft ? t('@sid (draft)', array('@sid' => $sid)) : $sid,
danielebarchiesi@0 48 format_date($submission->submitted, 'short'),
danielebarchiesi@0 49 );
danielebarchiesi@0 50 if (webform_results_access($node, $user)) {
danielebarchiesi@0 51 $row[] = theme('username', array('account' => $submission));
danielebarchiesi@0 52 $row[] = $submission->remote_addr;
danielebarchiesi@0 53 }
danielebarchiesi@0 54 $row[] = l(t('View'), "node/$node->nid/submission/$sid");
danielebarchiesi@0 55 $operation_count = 1;
danielebarchiesi@0 56 // No need to call this multiple times, just reference this in a variable.
danielebarchiesi@0 57 $destination = drupal_get_destination();
danielebarchiesi@0 58 if (webform_submission_access($node, $submission, 'edit', $user)) {
danielebarchiesi@0 59 $row[] = l(t('Edit'), "node/$node->nid/submission/$sid/edit", array('query' => $destination));
danielebarchiesi@0 60 $operation_count++;
danielebarchiesi@0 61 }
danielebarchiesi@0 62 if (webform_submission_access($node, $submission, 'delete', $user)) {
danielebarchiesi@0 63 $row[] = l(t('Delete'), "node/$node->nid/submission/$sid/delete", array('query' => $destination));
danielebarchiesi@0 64 $operation_count++;
danielebarchiesi@0 65 }
danielebarchiesi@0 66 if ($operation_count < $operation_total) {
danielebarchiesi@0 67 $row[count($row) - 1] = array('data' => $row[count($row) - 1], 'colspan' => $operation_total - $operation_count + 1);
danielebarchiesi@0 68 }
danielebarchiesi@0 69 $rows[] = $row;
danielebarchiesi@0 70 }
danielebarchiesi@0 71
danielebarchiesi@0 72 $element['#theme'] = 'webform_results_submissions';
danielebarchiesi@0 73 $element['#node'] = $node;
danielebarchiesi@0 74 $element['#submissions'] = $submissions;
danielebarchiesi@0 75 $element['#total_count'] = $count;
danielebarchiesi@0 76 $element['#pager_count'] = $pager_count;
danielebarchiesi@0 77 $element['#attached']['library'][] = array('webform', 'admin');
danielebarchiesi@0 78
danielebarchiesi@0 79 $element['table']['#theme'] = 'table';
danielebarchiesi@0 80 $element['table']['#header'] = $header;
danielebarchiesi@0 81 $element['table']['#rows'] = $rows;
danielebarchiesi@0 82 $element['table']['#operation_total'] = $operation_total;
danielebarchiesi@0 83
danielebarchiesi@0 84 return drupal_render($element);
danielebarchiesi@0 85
danielebarchiesi@0 86 }
danielebarchiesi@0 87
danielebarchiesi@0 88 /**
danielebarchiesi@0 89 * Theme the list of links for selecting the number of results per page.
danielebarchiesi@0 90 *
danielebarchiesi@0 91 * @param $total_count
danielebarchiesi@0 92 * The total number of results available.
danielebarchiesi@0 93 * @param $pager_count
danielebarchiesi@0 94 * The current number of results displayed per page.
danielebarchiesi@0 95 */
danielebarchiesi@0 96 function theme_webform_results_per_page($variables) {
danielebarchiesi@0 97 $total_count = $variables['total_count'];
danielebarchiesi@0 98 $pager_count = $variables['pager_count'];
danielebarchiesi@0 99 $output = '';
danielebarchiesi@0 100
danielebarchiesi@0 101 // Create a list of results-per-page options.
danielebarchiesi@0 102 $counts = array(
danielebarchiesi@0 103 '20' => '20',
danielebarchiesi@0 104 '50' => '50',
danielebarchiesi@0 105 '100' => '100',
danielebarchiesi@0 106 '200' => '200',
danielebarchiesi@0 107 '500' => '500',
danielebarchiesi@0 108 '1000' => '1000',
danielebarchiesi@0 109 '0' => t('All'),
danielebarchiesi@0 110 );
danielebarchiesi@0 111
danielebarchiesi@0 112 $count_links = array();
danielebarchiesi@0 113
danielebarchiesi@0 114 foreach ($counts as $number => $text) {
danielebarchiesi@0 115 if ($number < $total_count) {
danielebarchiesi@0 116 $count_links[] = l($text, $_GET['q'], array('query' => array('results' => $number), 'attributes' => array('class' => array($pager_count == $number ? 'selected' : ''))));
danielebarchiesi@0 117 }
danielebarchiesi@0 118 }
danielebarchiesi@0 119
danielebarchiesi@0 120 $output .= '<div class="webform-results-per-page">';
danielebarchiesi@0 121 if (count($count_links) > 1) {
danielebarchiesi@0 122 $output .= t('Show !count results per page.', array('!count' => implode(' | ', $count_links)));
danielebarchiesi@0 123 }
danielebarchiesi@0 124 else {
danielebarchiesi@0 125 $output .= t('Showing all results.');
danielebarchiesi@0 126 }
danielebarchiesi@0 127 if ($total_count > 1) {
danielebarchiesi@0 128 $output .= ' ' . t('@total results total.', array('@total' => $total_count));
danielebarchiesi@0 129 }
danielebarchiesi@0 130 $output .= '</div>';
danielebarchiesi@0 131
danielebarchiesi@0 132 return $output;
danielebarchiesi@0 133 }
danielebarchiesi@0 134
danielebarchiesi@0 135 /**
danielebarchiesi@0 136 * Theme the header of the submissions table.
danielebarchiesi@0 137 *
danielebarchiesi@0 138 * This is done in it's own function so that webform can retrieve the header and
danielebarchiesi@0 139 * use it for sorting the results.
danielebarchiesi@0 140 */
danielebarchiesi@0 141 function theme_webform_results_submissions_header($variables) {
danielebarchiesi@0 142 $node = $variables['node'];
danielebarchiesi@0 143
danielebarchiesi@0 144 $columns = array(
danielebarchiesi@0 145 array('data' => t('#'), 'field' => 'sid', 'sort' => 'desc'),
danielebarchiesi@0 146 array('data' => t('Submitted'), 'field' => 'submitted'),
danielebarchiesi@0 147 );
danielebarchiesi@0 148 if (webform_results_access($node)) {
danielebarchiesi@0 149 $columns[] = array('data' => t('User'), 'field' => 'name');
danielebarchiesi@0 150 $columns[] = array('data' => t('IP Address'), 'field' => 'remote_addr');
danielebarchiesi@0 151 }
danielebarchiesi@0 152 $columns[] = array('data' => t('Operations'), 'colspan' => module_exists('print') ? 5 : 3);
danielebarchiesi@0 153
danielebarchiesi@0 154 return $columns;
danielebarchiesi@0 155 }
danielebarchiesi@0 156
danielebarchiesi@0 157 /**
danielebarchiesi@0 158 * Preprocess function for webform-results-submissions.tpl.php
danielebarchiesi@0 159 */
danielebarchiesi@0 160 function template_preprocess_webform_results_submissions(&$vars) {
danielebarchiesi@0 161 $vars['node'] = $vars['element']['#node'];
danielebarchiesi@0 162 $vars['submissions'] = $vars['element']['#submissions'];
danielebarchiesi@0 163 $vars['table'] = $vars['element']['table'];
danielebarchiesi@0 164 $vars['total_count'] = $vars['element']['#total_count'];
danielebarchiesi@0 165 $vars['pager_count'] = $vars['element']['#pager_count'];
danielebarchiesi@0 166 $vars['is_submissions'] = (arg(2) == 'submissions')? 1 : 0;
danielebarchiesi@0 167
danielebarchiesi@0 168 unset($vars['element']);
danielebarchiesi@0 169 }
danielebarchiesi@0 170
danielebarchiesi@0 171 /**
danielebarchiesi@0 172 * Create a table containing all submitted values for a webform node.
danielebarchiesi@0 173 */
danielebarchiesi@0 174 function webform_results_table($node, $pager_count = 0) {
danielebarchiesi@0 175 if (isset($_GET['results']) && is_numeric($_GET['results'])) {
danielebarchiesi@0 176 $pager_count = $_GET['results'];
danielebarchiesi@0 177 }
danielebarchiesi@0 178
danielebarchiesi@0 179 // Get all the submissions for the node.
danielebarchiesi@0 180 $header = theme('webform_results_table_header', array('node' => $node));
danielebarchiesi@0 181 $submissions = webform_get_submissions($node->nid, $header, $pager_count);
danielebarchiesi@0 182 $total_count = webform_get_submission_count($node->nid);
danielebarchiesi@0 183
danielebarchiesi@0 184 $output = theme('webform_results_table', array('node' => $node, 'components' => $node->webform['components'], 'submissions' => $submissions, 'total_count' => $total_count, 'pager_count' => $pager_count));
danielebarchiesi@0 185 if ($pager_count) {
danielebarchiesi@0 186 $output .= theme('pager');
danielebarchiesi@0 187 }
danielebarchiesi@0 188 return $output;
danielebarchiesi@0 189 }
danielebarchiesi@0 190
danielebarchiesi@0 191 function theme_webform_results_table_header($variables) {
danielebarchiesi@0 192 return array(
danielebarchiesi@0 193 array('data' => t('#'), 'field' => 'sid', 'sort' => 'desc'),
danielebarchiesi@0 194 array('data' => t('Submitted'), 'field' => 'submitted'),
danielebarchiesi@0 195 array('data' => t('User'), 'field' => 'name'),
danielebarchiesi@0 196 array('data' => t('IP Address'), 'field' => 'remote_addr'),
danielebarchiesi@0 197 );
danielebarchiesi@0 198 }
danielebarchiesi@0 199
danielebarchiesi@0 200 /**
danielebarchiesi@0 201 * Theme the results table displaying all the submissions for a particular node.
danielebarchiesi@0 202 *
danielebarchiesi@0 203 * @param $node
danielebarchiesi@0 204 * The node whose results are being displayed.
danielebarchiesi@0 205 * @param $components
danielebarchiesi@0 206 * An associative array of the components for this webform.
danielebarchiesi@0 207 * @param $submissions
danielebarchiesi@0 208 * An array of all submissions for this webform.
danielebarchiesi@0 209 * @param $total_count
danielebarchiesi@0 210 * The total number of submissions to this webform.
danielebarchiesi@0 211 * @param $pager_count
danielebarchiesi@0 212 * The number of results to be shown per page.
danielebarchiesi@0 213 */
danielebarchiesi@0 214 function theme_webform_results_table($variables) {
danielebarchiesi@0 215 drupal_add_library('webform', 'admin');
danielebarchiesi@0 216
danielebarchiesi@0 217 $node = $variables['node'];
danielebarchiesi@0 218 $components = $variables['components'];
danielebarchiesi@0 219 $submissions = $variables['submissions'];
danielebarchiesi@0 220 $total_count = $variables['total_count'];
danielebarchiesi@0 221 $pager_count = $variables['pager_count'];
danielebarchiesi@0 222
danielebarchiesi@0 223 $header = array();
danielebarchiesi@0 224 $rows = array();
danielebarchiesi@0 225 $cell = array();
danielebarchiesi@0 226
danielebarchiesi@0 227 // This header has to be generated separately so we can add the SQL necessary.
danielebarchiesi@0 228 // to sort the results.
danielebarchiesi@0 229 $header = theme('webform_results_table_header', array('node' => $node));
danielebarchiesi@0 230
danielebarchiesi@0 231 // Generate a row for each submission.
danielebarchiesi@0 232 foreach ($submissions as $sid => $submission) {
danielebarchiesi@0 233 $cell[] = l($sid, 'node/' . $node->nid . '/submission/' . $sid);
danielebarchiesi@0 234 $cell[] = format_date($submission->submitted, 'short');
danielebarchiesi@0 235 $cell[] = theme('username', array('account' => $submission));
danielebarchiesi@0 236 $cell[] = $submission->remote_addr;
danielebarchiesi@0 237 $component_headers = array();
danielebarchiesi@0 238
danielebarchiesi@0 239 // Generate a cell for each component.
danielebarchiesi@0 240 foreach ($node->webform['components'] as $component) {
danielebarchiesi@0 241 $data = isset($submission->data[$component['cid']]['value']) ? $submission->data[$component['cid']]['value'] : NULL;
danielebarchiesi@0 242 $submission_output = webform_component_invoke($component['type'], 'table', $component, $data);
danielebarchiesi@0 243 if ($submission_output !== NULL) {
danielebarchiesi@0 244 $component_headers[] = check_plain($component['name']);
danielebarchiesi@0 245 $cell[] = $submission_output;
danielebarchiesi@0 246 }
danielebarchiesi@0 247 }
danielebarchiesi@0 248
danielebarchiesi@0 249 $rows[] = $cell;
danielebarchiesi@0 250 unset($cell);
danielebarchiesi@0 251 }
danielebarchiesi@0 252 if (!empty($component_headers)) {
danielebarchiesi@0 253 $header = array_merge($header, $component_headers);
danielebarchiesi@0 254 }
danielebarchiesi@0 255
danielebarchiesi@0 256 if (count($rows) == 0) {
danielebarchiesi@0 257 $rows[] = array(array('data' => t('There are no submissions for this form. <a href="!url">View this form</a>.', array('!url' => url('node/' . $node->nid))), 'colspan' => 4));
danielebarchiesi@0 258 }
danielebarchiesi@0 259
danielebarchiesi@0 260
danielebarchiesi@0 261 $output = '';
danielebarchiesi@0 262 $output .= theme('webform_results_per_page', array('total_count' => $total_count, 'pager_count' => $pager_count));
danielebarchiesi@0 263 $output .= theme('table', array('header' => $header, 'rows' => $rows));
danielebarchiesi@0 264 return $output;
danielebarchiesi@0 265 }
danielebarchiesi@0 266
danielebarchiesi@0 267 /**
danielebarchiesi@0 268 * Delete all submissions for a node.
danielebarchiesi@0 269 *
danielebarchiesi@0 270 * @param $nid
danielebarchiesi@0 271 * The node id whose submissions will be deleted.
danielebarchiesi@0 272 */
danielebarchiesi@0 273 function webform_results_clear($nid) {
danielebarchiesi@0 274 $node = node_load($nid);
danielebarchiesi@0 275 $submissions = webform_get_submissions($nid);
danielebarchiesi@0 276 foreach ($submissions as $submission) {
danielebarchiesi@0 277 webform_submission_delete($node, $submission);
danielebarchiesi@0 278 }
danielebarchiesi@0 279 }
danielebarchiesi@0 280
danielebarchiesi@0 281 /**
danielebarchiesi@0 282 * Confirmation form to delete all submissions for a node.
danielebarchiesi@0 283 *
danielebarchiesi@0 284 * @param $nid
danielebarchiesi@0 285 * ID of node for which to clear submissions.
danielebarchiesi@0 286 */
danielebarchiesi@0 287 function webform_results_clear_form($form, $form_state, $node) {
danielebarchiesi@0 288 drupal_set_title(t('Clear Form Submissions'));
danielebarchiesi@0 289
danielebarchiesi@0 290 $form = array();
danielebarchiesi@0 291 $form['nid'] = array('#type' => 'value', '#value' => $node->nid);
danielebarchiesi@0 292 $question = t('Are you sure you want to delete all submissions for this form?');
danielebarchiesi@0 293
danielebarchiesi@0 294 return confirm_form($form, $question, 'node/' . $node->nid . '/webform-results', NULL, t('Clear'), t('Cancel'));
danielebarchiesi@0 295 }
danielebarchiesi@0 296
danielebarchiesi@0 297 function webform_results_clear_form_submit($form, &$form_state) {
danielebarchiesi@0 298 webform_results_clear($form_state['values']['nid']);
danielebarchiesi@0 299 $node = node_load($form_state['values']['nid']);
danielebarchiesi@0 300 $title = $node->title;
danielebarchiesi@0 301 $message = t('Webform %title entries cleared.', array('%title' => $title));
danielebarchiesi@0 302 drupal_set_message($message);
danielebarchiesi@0 303 watchdog('webform', $message);
danielebarchiesi@0 304 $form_state['redirect'] = 'node/' . $form_state['values']['nid'] . '/webform-results';
danielebarchiesi@0 305 }
danielebarchiesi@0 306
danielebarchiesi@0 307 /**
danielebarchiesi@0 308 * Form to configure the download of CSV files.
danielebarchiesi@0 309 */
danielebarchiesi@0 310 function webform_results_download_form($form, &$form_state, $node) {
danielebarchiesi@0 311 module_load_include('inc', 'webform', 'includes/webform.export');
danielebarchiesi@0 312 module_load_include('inc', 'webform', 'includes/webform.components');
danielebarchiesi@0 313
danielebarchiesi@0 314 $form = array();
danielebarchiesi@0 315
danielebarchiesi@0 316 $form['node'] = array(
danielebarchiesi@0 317 '#type' => 'value',
danielebarchiesi@0 318 '#value' => $node,
danielebarchiesi@0 319 );
danielebarchiesi@0 320
danielebarchiesi@0 321 $form['format'] = array(
danielebarchiesi@0 322 '#type' => 'radios',
danielebarchiesi@0 323 '#title' => t('Export format'),
danielebarchiesi@0 324 '#options' => webform_export_list(),
danielebarchiesi@0 325 '#default_value' => isset($form_state['values']['format']) ? $form_state['values']['format'] : variable_get('webform_export_format', 'delimited'),
danielebarchiesi@0 326 );
danielebarchiesi@0 327
danielebarchiesi@0 328 $form['delimiter'] = array(
danielebarchiesi@0 329 '#type' => 'select',
danielebarchiesi@0 330 '#title' => t('Delimited text format'),
danielebarchiesi@0 331 '#description' => t('This is the delimiter used in the CSV/TSV file when downloading Webform results. Using tabs in the export is the most reliable method for preserving non-latin characters. You may want to change this to another character depending on the program with which you anticipate importing results.'),
danielebarchiesi@0 332 '#default_value' => isset($form_state['values']['delimiter']) ? $form_state['values']['delimiter'] : variable_get('webform_csv_delimiter', '\t'),
danielebarchiesi@0 333 '#options' => array(
danielebarchiesi@0 334 ',' => t('Comma (,)'),
danielebarchiesi@0 335 '\t' => t('Tab (\t)'),
danielebarchiesi@0 336 ';' => t('Semicolon (;)'),
danielebarchiesi@0 337 ':' => t('Colon (:)'),
danielebarchiesi@0 338 '|' => t('Pipe (|)'),
danielebarchiesi@0 339 '.' => t('Period (.)'),
danielebarchiesi@0 340 ' ' => t('Space ( )'),
danielebarchiesi@0 341 ),
danielebarchiesi@0 342 );
danielebarchiesi@0 343
danielebarchiesi@0 344 $form['select_options'] = array(
danielebarchiesi@0 345 '#type' => 'fieldset',
danielebarchiesi@0 346 '#title' => t('Select list options'),
danielebarchiesi@0 347 '#collapsible' => TRUE,
danielebarchiesi@0 348 '#collapsed' => TRUE,
danielebarchiesi@0 349 );
danielebarchiesi@0 350
danielebarchiesi@0 351 $form['select_options']['select_keys'] = array(
danielebarchiesi@0 352 '#type' => 'radios',
danielebarchiesi@0 353 '#title' => t('Select keys'),
danielebarchiesi@0 354 '#options' => array(
danielebarchiesi@0 355 0 => t('Full, human-readable options (values)'),
danielebarchiesi@0 356 1 => t('Short, raw options (keys)'),
danielebarchiesi@0 357 ),
danielebarchiesi@0 358 '#default_value' => isset($form_state['values']['select_options']['select_keys']) ? $form_state['values']['select_options']['select_keys'] : 0,
danielebarchiesi@0 359 '#description' => t('Choose which part of options should be displayed from key|value pairs.'),
danielebarchiesi@0 360 );
danielebarchiesi@0 361
danielebarchiesi@0 362 $form['select_options']['select_format'] = array(
danielebarchiesi@0 363 '#type' => 'radios',
danielebarchiesi@0 364 '#title' => t('Select list format'),
danielebarchiesi@0 365 '#options' => array(
danielebarchiesi@0 366 'separate' => t('Separate'),
danielebarchiesi@0 367 'compact' => t('Compact'),
danielebarchiesi@0 368 ),
danielebarchiesi@0 369 '#default_value' => isset($form_state['values']['select_options']['select_format']) ? $form_state['values']['select_options']['select_format'] : 'separate',
danielebarchiesi@0 370 '#attributes' => array('class' => array('webform-select-list-format')),
danielebarchiesi@0 371 '#theme' => 'webform_results_download_select_format',
danielebarchiesi@0 372 );
danielebarchiesi@0 373
danielebarchiesi@0 374 $csv_components = array(
danielebarchiesi@0 375 'info' => t('Submission information'),
danielebarchiesi@0 376 'serial' => '-' . t('Submission Number'),
danielebarchiesi@0 377 'sid' => '-' . t('Submission ID'),
danielebarchiesi@0 378 'time' => '-' . t('Time'),
danielebarchiesi@0 379 'draft' => '-' . t('Draft'),
danielebarchiesi@0 380 'ip_address' => '-' . t('IP Address'),
danielebarchiesi@0 381 'uid' => '-' . t('User ID'),
danielebarchiesi@0 382 'username' => '-' . t('Username'),
danielebarchiesi@0 383 );
danielebarchiesi@0 384 $csv_components += webform_component_list($node, 'csv', TRUE);
danielebarchiesi@0 385
danielebarchiesi@0 386 $form['components'] = array(
danielebarchiesi@0 387 '#type' => 'select',
danielebarchiesi@0 388 '#title' => t('Included export components'),
danielebarchiesi@0 389 '#options' => $csv_components,
danielebarchiesi@0 390 '#default_value' => isset($form_state['values']['components']) ? $form_state['values']['components'] : array_keys($csv_components),
danielebarchiesi@0 391 '#multiple' => TRUE,
danielebarchiesi@0 392 '#size' => 10,
danielebarchiesi@0 393 '#description' => t('The selected components will be included in the export.'),
danielebarchiesi@0 394 '#process' => array('webform_component_select'),
danielebarchiesi@0 395 );
danielebarchiesi@0 396
danielebarchiesi@0 397 $form['range'] = array(
danielebarchiesi@0 398 '#type' => 'fieldset',
danielebarchiesi@0 399 '#title' => t('Download range options'),
danielebarchiesi@0 400 '#collapsible' => TRUE,
danielebarchiesi@0 401 '#collapsed' => TRUE,
danielebarchiesi@0 402 '#tree' => TRUE,
danielebarchiesi@0 403 '#theme' => 'webform_results_download_range',
danielebarchiesi@0 404 '#element_validate' => array('webform_results_download_range_validate'),
danielebarchiesi@0 405 '#after_build' => array('webform_results_download_range_after_build'),
danielebarchiesi@0 406 );
danielebarchiesi@0 407
danielebarchiesi@0 408 $form['range']['range_type'] = array(
danielebarchiesi@0 409 '#type' => 'radios',
danielebarchiesi@0 410 '#options' => array(
danielebarchiesi@0 411 'all' => t('All submissions'),
danielebarchiesi@0 412 'new' => t('Only new submissions since your last download'),
danielebarchiesi@0 413 'latest' => t('Only the latest'),
danielebarchiesi@0 414 'range' => t('All submissions starting from'),
danielebarchiesi@0 415 ),
danielebarchiesi@0 416 '#default_value' => 'all',
danielebarchiesi@0 417 );
danielebarchiesi@0 418 $form['range']['latest'] = array(
danielebarchiesi@0 419 '#type' => 'textfield',
danielebarchiesi@0 420 '#size' => 5,
danielebarchiesi@0 421 '#maxlength' => 8,
danielebarchiesi@0 422 '#default_value' => isset($form_state['values']['latest']) ? $form_state['values']['latest'] : '',
danielebarchiesi@0 423 );
danielebarchiesi@0 424 $form['range']['start'] = array(
danielebarchiesi@0 425 '#type' => 'textfield',
danielebarchiesi@0 426 '#size' => 5,
danielebarchiesi@0 427 '#maxlength' => 8,
danielebarchiesi@0 428 '#default_value' => '',
danielebarchiesi@0 429 );
danielebarchiesi@0 430 $form['range']['end'] = array(
danielebarchiesi@0 431 '#type' => 'textfield',
danielebarchiesi@0 432 '#size' => 5,
danielebarchiesi@0 433 '#maxlength' => 8,
danielebarchiesi@0 434 '#default_value' => '',
danielebarchiesi@0 435 '#description' => '',
danielebarchiesi@0 436 );
danielebarchiesi@0 437
danielebarchiesi@0 438 // By default results are downloaded. User can override this value if
danielebarchiesi@0 439 // programmatically submitting this form.
danielebarchiesi@0 440 $form['download'] = array(
danielebarchiesi@0 441 '#type' => 'value',
danielebarchiesi@0 442 '#default_value' => TRUE
danielebarchiesi@0 443 );
danielebarchiesi@0 444
danielebarchiesi@0 445 $form['actions'] = array('#type' => 'actions');
danielebarchiesi@0 446 $form['actions']['submit'] = array(
danielebarchiesi@0 447 '#type' => 'submit',
danielebarchiesi@0 448 '#value' => t('Download'),
danielebarchiesi@0 449 );
danielebarchiesi@0 450
danielebarchiesi@0 451 return $form;
danielebarchiesi@0 452 }
danielebarchiesi@0 453
danielebarchiesi@0 454 /**
danielebarchiesi@0 455 * FormAPI element validate function for the range fieldset.
danielebarchiesi@0 456 */
danielebarchiesi@0 457 function webform_results_download_range_validate($element, $form_state) {
danielebarchiesi@0 458 switch ($element['range_type']['#value']) {
danielebarchiesi@0 459 case 'latest':
danielebarchiesi@0 460 // Download latest x submissions.
danielebarchiesi@0 461 if ($element['latest']['#value'] == '') {
danielebarchiesi@0 462 form_error($element['latest'], t('Latest number of submissions field is required.'));
danielebarchiesi@0 463 }
danielebarchiesi@0 464 else{
danielebarchiesi@0 465 if (!is_numeric($element['latest']['#value'])) {
danielebarchiesi@0 466 form_error($element['latest'], t('Latest number of submissions must be numeric.'));
danielebarchiesi@0 467 }
danielebarchiesi@0 468 else{
danielebarchiesi@0 469 if ($element['latest']['#value'] <= 0) {
danielebarchiesi@0 470 form_error($element['latest'], t('Latest number of submissions must be greater than 0.'));
danielebarchiesi@0 471 }
danielebarchiesi@0 472 }
danielebarchiesi@0 473 }
danielebarchiesi@0 474 break;
danielebarchiesi@0 475 case 'range':
danielebarchiesi@0 476 // Download Start-End range of submissions.
danielebarchiesi@0 477 // Start submission number.
danielebarchiesi@0 478 if ($element['start']['#value'] == '') {
danielebarchiesi@0 479 form_error($element['start'], t('Start submission number is required.'));
danielebarchiesi@0 480 }
danielebarchiesi@0 481 else{
danielebarchiesi@0 482 if (!is_numeric($element['start']['#value'])) {
danielebarchiesi@0 483 form_error($element['start'], t('Start submission number must be numeric.'));
danielebarchiesi@0 484 }
danielebarchiesi@0 485 else{
danielebarchiesi@0 486 if ($element['start']['#value'] <= 0) {
danielebarchiesi@0 487 form_error($element['start'], t('Start submission number must be greater than 0.'));
danielebarchiesi@0 488 }
danielebarchiesi@0 489 }
danielebarchiesi@0 490 }
danielebarchiesi@0 491 // End submission number.
danielebarchiesi@0 492 if ($element['end']['#value'] != '') {
danielebarchiesi@0 493 if (!is_numeric($element['end']['#value'])) {
danielebarchiesi@0 494 form_error($element['end'], t('End submission number must be numeric.'));
danielebarchiesi@0 495 }
danielebarchiesi@0 496 else{
danielebarchiesi@0 497 if ($element['end']['#value'] <= 0) {
danielebarchiesi@0 498 form_error($element['end'], t('End submission number must be greater than 0.'));
danielebarchiesi@0 499 }
danielebarchiesi@0 500 else{
danielebarchiesi@0 501 if ($element['end']['#value'] < $element['start']['#value']) {
danielebarchiesi@0 502 form_error($element['end'], t('End submission number may not be less than Start submission number.'));
danielebarchiesi@0 503 }
danielebarchiesi@0 504 }
danielebarchiesi@0 505 }
danielebarchiesi@0 506 }
danielebarchiesi@0 507 break;
danielebarchiesi@0 508 }
danielebarchiesi@0 509
danielebarchiesi@0 510 }
danielebarchiesi@0 511
danielebarchiesi@0 512 /**
danielebarchiesi@0 513 * Validate handler for webform_results_download_form().
danielebarchiesi@0 514 */
danielebarchiesi@0 515 function webform_results_download_form_submit(&$form, &$form_state) {
danielebarchiesi@0 516 $options = array(
danielebarchiesi@0 517 'delimiter' => $form_state['values']['delimiter'],
danielebarchiesi@0 518 'components' => array_keys(array_filter($form_state['values']['components'])),
danielebarchiesi@0 519 'select_keys' => $form_state['values']['select_keys'],
danielebarchiesi@0 520 'select_format' => $form_state['values']['select_format'],
danielebarchiesi@0 521 'range_type' => $form_state['values']['range']['range_type'],
danielebarchiesi@0 522 'download' => $form_state['values']['download'],
danielebarchiesi@0 523 );
danielebarchiesi@0 524
danielebarchiesi@0 525 // Retrieve the list of required SIDs.
danielebarchiesi@0 526 if ($options['range_type'] != 'all') {
danielebarchiesi@0 527 $options['sids'] = webform_download_sids($form_state['values']['node']->nid, $form_state['values']['range']);
danielebarchiesi@0 528 }
danielebarchiesi@0 529
danielebarchiesi@0 530 $export_info = webform_results_export($form_state['values']['node'], $form_state['values']['format'], $options);
danielebarchiesi@0 531
danielebarchiesi@0 532 // If webform result file should be downloaded, send the file to the browser,
danielebarchiesi@0 533 // otherwise save information about the created file in $form_state.
danielebarchiesi@0 534 if ($options['download']) {
danielebarchiesi@0 535 webform_results_download($form_state['values']['node'], $export_info);
danielebarchiesi@0 536 }
danielebarchiesi@0 537 else {
danielebarchiesi@0 538 $form_state['export_info'] = $export_info;
danielebarchiesi@0 539 }
danielebarchiesi@0 540 }
danielebarchiesi@0 541
danielebarchiesi@0 542 /**
danielebarchiesi@0 543 * FormAPI after build function for the download range fieldset.
danielebarchiesi@0 544 */
danielebarchiesi@0 545 function webform_results_download_range_after_build($element, &$form_state) {
danielebarchiesi@0 546 $node = $form_state['values']['node'];
danielebarchiesi@0 547
danielebarchiesi@0 548 // Build a list of counts of new and total submissions.
danielebarchiesi@0 549 $count = webform_get_submission_count($node->nid);
danielebarchiesi@0 550 $sids = webform_download_sids($node->nid, array('range_type' => 'new'));
danielebarchiesi@0 551
danielebarchiesi@0 552 $last_download = webform_download_last_download_info($node->nid);
danielebarchiesi@0 553
danielebarchiesi@0 554 $element['#webform_download_info']['sid'] = $last_download ? $last_download['sid'] : 0;
danielebarchiesi@0 555 $element['#webform_download_info']['requested'] = $last_download ? $last_download['requested'] : $node->created;
danielebarchiesi@0 556 $element['#webform_download_info']['total'] = $count;
danielebarchiesi@0 557 $element['#webform_download_info']['new'] = count($sids);
danielebarchiesi@0 558
danielebarchiesi@0 559 return $element;
danielebarchiesi@0 560 }
danielebarchiesi@0 561
danielebarchiesi@0 562 /**
danielebarchiesi@0 563 * Theme the output of the export range fieldset.
danielebarchiesi@0 564 */
danielebarchiesi@0 565 function theme_webform_results_download_range($variables) {
danielebarchiesi@0 566 drupal_add_library('webform', 'admin');
danielebarchiesi@0 567
danielebarchiesi@0 568 $element = $variables['element'];
danielebarchiesi@0 569 $download_info = $element['#webform_download_info'];
danielebarchiesi@0 570
danielebarchiesi@0 571 // Set description for total of all submissions.
danielebarchiesi@0 572 $element['range_type']['all']['#theme_wrappers'] = array('webform_inline_radio');
danielebarchiesi@0 573 $element['range_type']['all']['#description'] = '(' . t('@count total', array('@count' => $download_info['total'])) . ')';
danielebarchiesi@0 574
danielebarchiesi@0 575 // Set description for "New submissions since last download".
danielebarchiesi@0 576 $format = webform_date_format('short');
danielebarchiesi@0 577 $requested_date = format_date($download_info['requested'], 'custom', $format);
danielebarchiesi@0 578 $element['range_type']['new']['#theme_wrappers'] = array('webform_inline_radio');
danielebarchiesi@0 579 $element['range_type']['new']['#description'] = '(' . t('@count new since @date', array('@count' => $download_info['new'], '@date' => $requested_date)) . ')';
danielebarchiesi@0 580
danielebarchiesi@0 581
danielebarchiesi@0 582 // Disable option if there are no new submissions.
danielebarchiesi@0 583 if ($download_info['new'] == 0) {
danielebarchiesi@0 584 $element['range_type']['new']['#attributes']['disabled'] = 'disabled';
danielebarchiesi@0 585 }
danielebarchiesi@0 586
danielebarchiesi@0 587 // Render latest x submissions option.
danielebarchiesi@0 588 $element['latest']['#attributes']['class'] = array('webform-set-active');
danielebarchiesi@0 589 $element['range_type']['latest']['#theme_wrappers'] = array('webform_inline_radio');
danielebarchiesi@0 590 $element['range_type']['latest']['#inline_element'] = t('Only the latest !number submissions', array('!number' => drupal_render($element['latest'])));
danielebarchiesi@0 591 $element['range_type']['latest']['#title'] = NULL;
danielebarchiesi@0 592
danielebarchiesi@0 593 // Render Start-End submissions option.
danielebarchiesi@0 594 $element['start']['#attributes']['class'] = array('webform-set-active');
danielebarchiesi@0 595 $element['end']['#attributes']['class'] = array('webform-set-active');
danielebarchiesi@0 596 $element['range_type']['range']['#theme_wrappers'] = array('webform_inline_radio');
danielebarchiesi@0 597 $element['range_type']['range']['#inline_element'] = t('All submissions starting from: !start and optionally to: !end', array('!start' => drupal_render($element['start']), '!end' => drupal_render($element['end'])));
danielebarchiesi@0 598 $element['range_type']['range']['#title'] = NULL;
danielebarchiesi@0 599
danielebarchiesi@0 600 $last_sid = $download_info['sid'] ? $download_info['sid'] : drupal_placeholder(t('none'));
danielebarchiesi@0 601 $element['range_type']['range']['#description'] = '(' . t('Use submission IDs for the range. Last downloaded end SID: !sid.', array('!sid' => $last_sid)) . ')';
danielebarchiesi@0 602
danielebarchiesi@0 603 return drupal_render_children($element);
danielebarchiesi@0 604 }
danielebarchiesi@0 605
danielebarchiesi@0 606 /**
danielebarchiesi@0 607 * Theme the output of the select list format radio buttons.
danielebarchiesi@0 608 */
danielebarchiesi@0 609 function theme_webform_results_download_select_format($variables) {
danielebarchiesi@0 610 drupal_add_library('webform', 'admin');
danielebarchiesi@0 611
danielebarchiesi@0 612 $element = $variables['element'];
danielebarchiesi@0 613 $output = '';
danielebarchiesi@0 614
danielebarchiesi@0 615 // Build an example table for the separate option.
danielebarchiesi@0 616 $header = array(t('Option A'), t('Option B'), t('Option C'));
danielebarchiesi@0 617 $rows = array(
danielebarchiesi@0 618 array('X', '', ''),
danielebarchiesi@0 619 array('X', '', 'X'),
danielebarchiesi@0 620 array('', 'X', 'X'),
danielebarchiesi@0 621 );
danielebarchiesi@0 622
danielebarchiesi@0 623 $element['separate']['#attributes']['class'] = array();
danielebarchiesi@0 624 $element['separate']['#description'] = theme('table', array('header' => $header, 'rows' => $rows));
danielebarchiesi@0 625 $element['separate']['#description'] .= t('Separate options are more suitable for building reports, graphs, and statistics in a spreadsheet application.');
danielebarchiesi@0 626 $output .= drupal_render($element['separate']);
danielebarchiesi@0 627
danielebarchiesi@0 628 // Build an example table for the compact option.
danielebarchiesi@0 629 $header = array(t('My select list'));
danielebarchiesi@0 630 $rows = array(
danielebarchiesi@0 631 array('Option A'),
danielebarchiesi@0 632 array('Option A,Option C'),
danielebarchiesi@0 633 array('Option B,Option C'),
danielebarchiesi@0 634 );
danielebarchiesi@0 635
danielebarchiesi@0 636 $element['compact']['#attributes']['class'] = array();
danielebarchiesi@0 637 $element['compact']['#description'] = theme('table', array('header' => $header, 'rows' => $rows));
danielebarchiesi@0 638 $element['compact']['#description'] .= t('Compact options are more suitable for importing data into other systems.');
danielebarchiesi@0 639 $output .= drupal_render($element['compact']);
danielebarchiesi@0 640
danielebarchiesi@0 641 return $output;
danielebarchiesi@0 642 }
danielebarchiesi@0 643
danielebarchiesi@0 644 /**
danielebarchiesi@0 645 * Generate a Excel-readable CSV file containing all submissions for a Webform.
danielebarchiesi@0 646 *
danielebarchiesi@0 647 * The CSV requires that the data be presented in a flat file. In order
danielebarchiesi@0 648 * to maximize usability to the Excel community and minimize subsequent
danielebarchiesi@0 649 * stats or spreadsheet programming this program extracts data from the
danielebarchiesi@0 650 * various records for a given session and presents them as a single file
danielebarchiesi@0 651 * where each row represents a single record.
danielebarchiesi@0 652 * The structure of the file is:
danielebarchiesi@0 653 * Heading Line 1: Gives group overviews padded by empty cells to the
danielebarchiesi@0 654 * next group. A group may be a question and corresponds
danielebarchiesi@0 655 * to a component in the webform philosophy. Each group
danielebarchiesi@0 656 * overview will have a fixed number of columns beneath it.
danielebarchiesi@0 657 * Heading line 2: gives column headings
danielebarchiesi@0 658 * Data line 1 .....
danielebarchiesi@0 659 * Data line 2 .....
danielebarchiesi@0 660 *
danielebarchiesi@0 661 * An example of this format is given below. Note the columns have had spaces
danielebarchiesi@0 662 * added so the columns line up. This is not the case with actual file where
danielebarchiesi@0 663 * a column may be null. Note also, that multiple choice questions as produced
danielebarchiesi@0 664 * by checkboxes or radio buttons have been presented as "yes" or "no" and the
danielebarchiesi@0 665 * actual choice text is retained only in the header line 2.
danielebarchiesi@0 666 * Data from text boxes and input fields are written out in the body of the table.
danielebarchiesi@0 667 *
danielebarchiesi@0 668 * Submission Details, , , ,Question 1, , ,.., ,Question 2, , ,.., ,Question n
danielebarchiesi@0 669 * timestamp ,time,SID,userid,Choice 1 ,Choice 2,Choice 3,..,Choice n,Choice 1 ,Choice 2,Choice 3,..,Choice n,Comment
danielebarchiesi@0 670 * 21 Feb 2005 ,1835,23 ,34 ,X , , ,.., ,X ,X ,X ,..,X ,My comment
danielebarchiesi@0 671 * 23 Feb 2005 ,1125,24 ,89 ,X ,X , ,.., ,X ,X ,X ,..,X ,Hello
danielebarchiesi@0 672 * .................................................................................................................................
danielebarchiesi@0 673 * 27 Feb 2005 ,1035,56 ,212 ,X , , ,.., ,X ,X ,X ,..,X ,How is this?
danielebarchiesi@0 674 *
danielebarchiesi@0 675 */
danielebarchiesi@0 676 function webform_results_export($node, $format = 'delimited', $options = array()) {
danielebarchiesi@0 677 global $user;
danielebarchiesi@0 678 module_load_include('inc', 'webform', 'includes/webform.export');
danielebarchiesi@0 679 module_load_include('inc', 'webform', 'includes/webform.components');
danielebarchiesi@0 680
danielebarchiesi@0 681 $submission_information = array(
danielebarchiesi@0 682 'serial' => t('Serial'),
danielebarchiesi@0 683 'sid' => t('SID'),
danielebarchiesi@0 684 'time' => t('Time'),
danielebarchiesi@0 685 'draft' => t('Draft'),
danielebarchiesi@0 686 'ip_address' => t('IP Address'),
danielebarchiesi@0 687 'uid' => t('UID'),
danielebarchiesi@0 688 'username' => t('Username'),
danielebarchiesi@0 689 );
danielebarchiesi@0 690
danielebarchiesi@0 691 if (empty($options)) {
danielebarchiesi@0 692 $options = array(
danielebarchiesi@0 693 'delimiter' => variable_get('webform_csv_delimiter', '\t'),
danielebarchiesi@0 694 'components' => array_merge(array_keys($submission_information), array_keys(webform_component_list($node, 'csv', TRUE))),
danielebarchiesi@0 695 'select_keys' => 0,
danielebarchiesi@0 696 'select_format' => 'separate',
danielebarchiesi@0 697 'range_type' => 'all',
danielebarchiesi@0 698 );
danielebarchiesi@0 699 }
danielebarchiesi@0 700 else {
danielebarchiesi@0 701 foreach ($submission_information as $key => $label) {
danielebarchiesi@0 702 if (!in_array($key, $options['components'])) {
danielebarchiesi@0 703 unset($submission_information[$key]);
danielebarchiesi@0 704 }
danielebarchiesi@0 705 }
danielebarchiesi@0 706 }
danielebarchiesi@0 707
danielebarchiesi@0 708 // Open a new Webform exporter object.
danielebarchiesi@0 709 $exporter = webform_export_create_handler($format, $options);
danielebarchiesi@0 710
danielebarchiesi@0 711 $file_name = drupal_tempnam('temporary://', 'webform_');
danielebarchiesi@0 712 $handle = @fopen($file_name, 'w'); // The @ suppresses errors.
danielebarchiesi@0 713 $exporter->bof($handle);
danielebarchiesi@0 714
danielebarchiesi@0 715 // Fill in the header for the submission information (if any).
danielebarchiesi@0 716 $header[2] = $header[1] = $header[0] = count($submission_information) ? array_fill(0, count($submission_information), '') : array();
danielebarchiesi@0 717 if (count($submission_information)) {
danielebarchiesi@0 718 $header[0][0] = $node->title;
danielebarchiesi@0 719 $header[1][0] = t('Submission Details');
danielebarchiesi@0 720 foreach (array_values($submission_information) as $column => $label) {
danielebarchiesi@0 721 $header[2][$column] = $label;
danielebarchiesi@0 722 }
danielebarchiesi@0 723 }
danielebarchiesi@0 724
danielebarchiesi@0 725 // Compile header information for components.
danielebarchiesi@0 726 foreach ($options['components'] as $cid) {
danielebarchiesi@0 727 if (isset($node->webform['components'][$cid])) {
danielebarchiesi@0 728 $component = $node->webform['components'][$cid];
danielebarchiesi@0 729
danielebarchiesi@0 730 // Let each component determine its headers.
danielebarchiesi@0 731 if (webform_component_feature($component['type'], 'csv')) {
danielebarchiesi@0 732 $component_header = (array) webform_component_invoke($component['type'], 'csv_headers', $component, $options);
danielebarchiesi@0 733 $header[0] = array_merge($header[0], (array) $component_header[0]);
danielebarchiesi@0 734 $header[1] = array_merge($header[1], (array) $component_header[1]);
danielebarchiesi@0 735 $header[2] = array_merge($header[2], (array) $component_header[2]);
danielebarchiesi@0 736 }
danielebarchiesi@0 737 }
danielebarchiesi@0 738 }
danielebarchiesi@0 739
danielebarchiesi@0 740 // Add headers to the file.
danielebarchiesi@0 741 foreach ($header as $row) {
danielebarchiesi@0 742 $exporter->add_row($handle, $row);
danielebarchiesi@0 743 }
danielebarchiesi@0 744
danielebarchiesi@0 745 // Get all the required submissions for the download.
danielebarchiesi@0 746 $filters['nid'] = $node->nid;
danielebarchiesi@0 747 if (!empty($options['sids'])){
danielebarchiesi@0 748 $filters['sid'] = $options['sids'];
danielebarchiesi@0 749 }
danielebarchiesi@0 750 $submissions = webform_get_submissions($filters);
danielebarchiesi@0 751
danielebarchiesi@0 752 // Generate a row for each submission.
danielebarchiesi@0 753 $row_count = 0;
danielebarchiesi@0 754 foreach ($submissions as $sid => $submission) {
danielebarchiesi@0 755 $row_count++;
danielebarchiesi@0 756
danielebarchiesi@0 757 $row = array();
danielebarchiesi@0 758 if (isset($submission_information['serial'])) {
danielebarchiesi@0 759 $row[] = $row_count;
danielebarchiesi@0 760 }
danielebarchiesi@0 761 if (isset($submission_information['sid'])) {
danielebarchiesi@0 762 $row[] = $sid;
danielebarchiesi@0 763 }
danielebarchiesi@0 764 if (isset($submission_information['time'])) {
danielebarchiesi@0 765 $row[] = format_date($submission->submitted, 'short');
danielebarchiesi@0 766 }
danielebarchiesi@0 767 if (isset($submission_information['draft'])) {
danielebarchiesi@0 768 $row[] = $submission->is_draft;
danielebarchiesi@0 769 }
danielebarchiesi@0 770 if (isset($submission_information['ip_address'])) {
danielebarchiesi@0 771 $row[] = $submission->remote_addr;
danielebarchiesi@0 772 }
danielebarchiesi@0 773 if (isset($submission_information['uid'])) {
danielebarchiesi@0 774 $row[] = $submission->uid;
danielebarchiesi@0 775 }
danielebarchiesi@0 776 if (isset($submission_information['username'])) {
danielebarchiesi@0 777 $row[] = $submission->name;
danielebarchiesi@0 778 }
danielebarchiesi@0 779
danielebarchiesi@0 780 foreach ($options['components'] as $cid) {
danielebarchiesi@0 781 if (isset($node->webform['components'][$cid])) {
danielebarchiesi@0 782 $component = $node->webform['components'][$cid];
danielebarchiesi@0 783 // Let each component add its data.
danielebarchiesi@0 784 $raw_data = isset($submission->data[$cid]['value']) ? $submission->data[$cid]['value'] : NULL;
danielebarchiesi@0 785 if (webform_component_feature($component['type'], 'csv')) {
danielebarchiesi@0 786 $data = webform_component_invoke($component['type'], 'csv_data', $component, $options, $raw_data);
danielebarchiesi@0 787 if (is_array($data)) {
danielebarchiesi@0 788 $row = array_merge($row, array_values($data));
danielebarchiesi@0 789 }
danielebarchiesi@0 790 else {
danielebarchiesi@0 791 $row[] = isset($data) ? $data : '';
danielebarchiesi@0 792 }
danielebarchiesi@0 793 }
danielebarchiesi@0 794 }
danielebarchiesi@0 795 }
danielebarchiesi@0 796
danielebarchiesi@0 797 // Write data from submissions.
danielebarchiesi@0 798 $data = $exporter->add_row($handle, $row);
danielebarchiesi@0 799 }
danielebarchiesi@0 800
danielebarchiesi@0 801 // Add the closing bytes.
danielebarchiesi@0 802 $exporter->eof($handle);
danielebarchiesi@0 803
danielebarchiesi@0 804 // Close the file.
danielebarchiesi@0 805 @fclose($handle);
danielebarchiesi@0 806
danielebarchiesi@0 807 $export_info['options'] = $options;
danielebarchiesi@0 808 $export_info['file_name'] = $file_name;
danielebarchiesi@0 809 $export_info['exporter'] = $exporter;
danielebarchiesi@0 810 $export_info['row_count'] = $row_count;
danielebarchiesi@0 811 $export_info['last_sid'] = $sid;
danielebarchiesi@0 812
danielebarchiesi@0 813 return $export_info;
danielebarchiesi@0 814 }
danielebarchiesi@0 815
danielebarchiesi@0 816 /**
danielebarchiesi@0 817 * Send a generated webform results file to the user's browser.
danielebarchiesi@0 818 *
danielebarchiesi@0 819 * @param $node
danielebarchiesi@0 820 * The webform node.
danielebarchiesi@0 821 * @param $export_info
danielebarchiesi@0 822 * Export information array retrieved from webform_results_export().
danielebarchiesi@0 823 */
danielebarchiesi@0 824 function webform_results_download($node, $export_info) {
danielebarchiesi@0 825 global $user;
danielebarchiesi@0 826
danielebarchiesi@0 827 // $exporter, $file_name, $row_count
danielebarchiesi@0 828 $export_name = _webform_safe_name($node->title);
danielebarchiesi@0 829 $export_info['exporter']->set_headers($export_name);
danielebarchiesi@0 830 @readfile($export_info['file_name']); // The @ makes it silent.
danielebarchiesi@0 831 @unlink($export_info['file_name']); // Clean up, the @ makes it silent.
danielebarchiesi@0 832
danielebarchiesi@0 833 // Update user last downloaded sid if required.
danielebarchiesi@0 834 if ($export_info['options']['range_type'] != 'range' && $export_info['row_count'] > 0) {
danielebarchiesi@0 835 // Delete existing record.
danielebarchiesi@0 836 db_delete('webform_last_download')
danielebarchiesi@0 837 ->condition('nid', $node->nid)
danielebarchiesi@0 838 ->condition('uid', $user->uid)
danielebarchiesi@0 839 ->execute();
danielebarchiesi@0 840 // Write new record.
danielebarchiesi@0 841 db_insert('webform_last_download')
danielebarchiesi@0 842 ->fields(array(
danielebarchiesi@0 843 'nid' => $node->nid,
danielebarchiesi@0 844 'uid' => $user->uid,
danielebarchiesi@0 845 'sid' => $export_info['last_sid'],
danielebarchiesi@0 846 'requested' => REQUEST_TIME,
danielebarchiesi@0 847 ))
danielebarchiesi@0 848 ->execute();
danielebarchiesi@0 849 }
danielebarchiesi@0 850
danielebarchiesi@0 851 exit();
danielebarchiesi@0 852 }
danielebarchiesi@0 853
danielebarchiesi@0 854 /**
danielebarchiesi@0 855 * Provides a simple analysis of all submissions to a webform.
danielebarchiesi@0 856 *
danielebarchiesi@0 857 * @param $node
danielebarchiesi@0 858 * The webform node on which to generate the analysis.
danielebarchiesi@0 859 * @param $sids
danielebarchiesi@0 860 * An array of submission IDs to which this analysis may be filtered. May be
danielebarchiesi@0 861 * used to generate results that are per-user or other groups of submissions.
danielebarchiesi@0 862 * @param $analysis_component
danielebarchiesi@0 863 * A webform component. If passed in, additional information may be returned
danielebarchiesi@0 864 * relating specifically to that component's analysis, such as a list of
danielebarchiesi@0 865 * "Other" values within a select list.
danielebarchiesi@0 866 */
danielebarchiesi@0 867 function webform_results_analysis($node, $sids = array(), $analysis_component = NULL) {
danielebarchiesi@0 868 if (!is_array($sids)) {
danielebarchiesi@0 869 $sids = array();
danielebarchiesi@0 870 }
danielebarchiesi@0 871
danielebarchiesi@0 872 // If showing a component's details, we don't want to loose the menu tabs.
danielebarchiesi@0 873 if ($analysis_component) {
danielebarchiesi@0 874 $item = menu_get_item('node/' . $node->nid . '/webform-results/analysis');
danielebarchiesi@0 875 menu_set_item(NULL, $item);
danielebarchiesi@0 876 }
danielebarchiesi@0 877
danielebarchiesi@0 878 $components = isset($analysis_component) ? array($analysis_component['cid'] => $analysis_component) : $node->webform['components'];
danielebarchiesi@0 879 $data = array();
danielebarchiesi@0 880 foreach ($components as $cid => $component) {
danielebarchiesi@0 881 // Do component specific call.
danielebarchiesi@0 882 if ($row_data = webform_component_invoke($component['type'], 'analysis', $component, $sids, isset($analysis_component))) {
danielebarchiesi@0 883 $data[$cid] = $row_data;
danielebarchiesi@0 884 }
danielebarchiesi@0 885 }
danielebarchiesi@0 886
danielebarchiesi@0 887 return theme('webform_results_analysis', array('node' => $node, 'data' => $data, 'sids' => $sids, 'component' => $analysis_component));
danielebarchiesi@0 888 }
danielebarchiesi@0 889
danielebarchiesi@0 890 /**
danielebarchiesi@0 891 * Output the content of the Analysis page.
danielebarchiesi@0 892 *
danielebarchiesi@0 893 * @see webform_results_analysis()
danielebarchiesi@0 894 */
danielebarchiesi@0 895 function theme_webform_results_analysis($variables) {
danielebarchiesi@0 896 $node = $variables['node'];
danielebarchiesi@0 897 $data = $variables['data'];
danielebarchiesi@0 898 $sids = $variables['sids'];
danielebarchiesi@0 899 $analysis_component = $variables['component'];
danielebarchiesi@0 900
danielebarchiesi@0 901 $rows = array();
danielebarchiesi@0 902 $question_number = 0;
danielebarchiesi@0 903 $single = isset($analysis_component);
danielebarchiesi@0 904
danielebarchiesi@0 905 $header = array(
danielebarchiesi@0 906 $single ? $analysis_component['name'] : t('Q'),
danielebarchiesi@0 907 array('data' => $single ? '&nbsp;' : t('responses'), 'colspan' => '10')
danielebarchiesi@0 908 );
danielebarchiesi@0 909
danielebarchiesi@0 910 foreach ($data as $cid => $row_data) {
danielebarchiesi@0 911 $question_number++;
danielebarchiesi@0 912
danielebarchiesi@0 913 if (is_array($row_data)) {
danielebarchiesi@0 914 $row = array();
danielebarchiesi@0 915 if (!$single) {
danielebarchiesi@0 916 $row['data'][] = array('data' => '<strong>' . $question_number . '</strong>', 'rowspan' => count($row_data) + 1, 'valign' => 'top');
danielebarchiesi@0 917 $row['data'][] = array('data' => '<strong>' . check_plain($node->webform['components'][$cid]['name']) . '</strong>', 'colspan' => '10');
danielebarchiesi@0 918 $row['class'][] = 'webform-results-question';
danielebarchiesi@0 919 }
danielebarchiesi@0 920 $rows = array_merge($rows, array_merge(array($row), $row_data));
danielebarchiesi@0 921 }
danielebarchiesi@0 922 }
danielebarchiesi@0 923
danielebarchiesi@0 924 if (count($rows) == 0) {
danielebarchiesi@0 925 $rows[] = array(array('data' => t('There are no submissions for this form. <a href="!url">View this form</a>.', array('!url' => url('node/' . $node->nid))), 'colspan' => 20));
danielebarchiesi@0 926 }
danielebarchiesi@0 927
danielebarchiesi@0 928 return theme('table', array('header' => $header, 'rows' => $rows, 'attributes' => array('class' => array('webform-results-analysis'))));
danielebarchiesi@0 929 }
danielebarchiesi@0 930
danielebarchiesi@0 931 /**
danielebarchiesi@0 932 * Given a set of range options, retrieve a set of SIDs for a webform node.
danielebarchiesi@0 933 */
danielebarchiesi@0 934 function webform_download_sids($nid, $range_options, $uid = NULL) {
danielebarchiesi@0 935 $query = db_select('webform_submissions', 'ws')
danielebarchiesi@0 936 ->fields('ws', array('sid'))
danielebarchiesi@0 937 ->condition('nid', $nid);
danielebarchiesi@0 938
danielebarchiesi@0 939 switch ($range_options['range_type']) {
danielebarchiesi@0 940 case 'all':
danielebarchiesi@0 941 // All Submissions.
danielebarchiesi@0 942 $query->orderBy('sid', 'ASC');
danielebarchiesi@0 943 break;
danielebarchiesi@0 944 case 'new':
danielebarchiesi@0 945 // All Since Last Download.
danielebarchiesi@0 946 $download_info = webform_download_last_download_info($nid, $uid);
danielebarchiesi@0 947 $last_sid = $download_info ? $download_info['sid'] : 0;
danielebarchiesi@0 948 $query
danielebarchiesi@0 949 ->condition('sid', $last_sid, '>')
danielebarchiesi@0 950 ->orderBy('sid', 'ASC');
danielebarchiesi@0 951 break;
danielebarchiesi@0 952 case 'latest':
danielebarchiesi@0 953 // Last x Submissions.
danielebarchiesi@0 954 $query
danielebarchiesi@0 955 ->orderBy('sid', 'DESC')
danielebarchiesi@0 956 ->range(0, $range_options['latest']);
danielebarchiesi@0 957 break;
danielebarchiesi@0 958 case 'range':
danielebarchiesi@0 959 // Submissions Start-End.
danielebarchiesi@0 960 $query->condition('sid', $range_options['start'], '>=');
danielebarchiesi@0 961 if ($range_options['end']){
danielebarchiesi@0 962 $query->condition('sid', $range_options['end'], '<=');
danielebarchiesi@0 963 }
danielebarchiesi@0 964 $query->orderBy('sid', 'ASC');
danielebarchiesi@0 965 break;
danielebarchiesi@0 966 }
danielebarchiesi@0 967
danielebarchiesi@0 968 $sids = $query->execute()->fetchCol();
danielebarchiesi@0 969
danielebarchiesi@0 970 // The last x submissions option has SIDs that are in reverse order.
danielebarchiesi@0 971 if ($range_options['range_type'] == 'latest') {
danielebarchiesi@0 972 $sids = array_reverse($sids);
danielebarchiesi@0 973 }
danielebarchiesi@0 974
danielebarchiesi@0 975 return $sids;
danielebarchiesi@0 976 }
danielebarchiesi@0 977
danielebarchiesi@0 978 /**
danielebarchiesi@0 979 * Get this user's last download information, including the SID and timestamp.
danielebarchiesi@0 980 *
danielebarchiesi@0 981 * This function provides an array of information about the last download that
danielebarchiesi@0 982 * a user had for a particular Webform node. Currently it only returns an array
danielebarchiesi@0 983 * with two keys:
danielebarchiesi@0 984 * - sid: The last submission ID that was downloaded.
danielebarchiesi@0 985 * - requested: The timestamp of the last download request.
danielebarchiesi@0 986 *
danielebarchiesi@0 987 * @param $nid
danielebarchiesi@0 988 * The Webform NID.
danielebarchiesi@0 989 * @param $uid
danielebarchiesi@0 990 * The user account ID for which to retrieve download information.
danielebarchiesi@0 991 * @return
danielebarchiesi@0 992 * An array of download information or FALSE if this user has never downloaded
danielebarchiesi@0 993 * results for this particular node.
danielebarchiesi@0 994 */
danielebarchiesi@0 995 function webform_download_last_download_info($nid, $uid = NULL) {
danielebarchiesi@0 996 $uid = isset($uid) ? $uid : $GLOBALS['user']->uid;
danielebarchiesi@0 997
danielebarchiesi@0 998 $info = db_select('webform_last_download', 'wld')
danielebarchiesi@0 999 ->fields('wld')
danielebarchiesi@0 1000 ->condition('nid', $nid)
danielebarchiesi@0 1001 ->condition('uid', $uid)
danielebarchiesi@0 1002 ->execute()
danielebarchiesi@0 1003 ->fetchAssoc();
danielebarchiesi@0 1004
danielebarchiesi@0 1005 return $info;
danielebarchiesi@0 1006 }