'',
'form_key' => NULL,
'email' => 1,
'mandatory' => 0,
'pid' => 0,
'weight' => 0,
'value' => '',
'extra' => array(
'items' => '',
'email' => 0,
'multiple' => NULL,
'aslist' => NULL,
'description' => '',
),
);
}
/**
* Create a set of form items to be displayed on the form for editing this
* component. Use care naming the form items, as this correlates directly to the
* database schema. The component "Name" and "Description" fields are added to
* every component type and are not necessary to specify here (although they may
* be overridden if desired).
* @return
* An array of form items to be displayed on the edit component page.
*/
function _webform_edit_select($currfield) {
$edit_fields = array();
$edit_fields['extra']['items'] = array(
'#type' => 'textarea',
'#title' => t('Options'),
'#default_value' => $currfield['extra']['items'],
'#description' => t('A list of selectable options. One option per line. Key-value pairs may be entered seperated by pipes, such as "safe_key|Some readable option". Option groups for lists and menus 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'),
'#cols' => 60,
'#rows' => 5,
'#weight' => -2,
'#required' => TRUE,
'#element_validate' => array('_webform_edit_validate_select'),
);
$edit_fields['value'] = array(
'#type' => 'textfield',
'#title' => t('Default value'),
'#default_value' => $currfield['value'],
'#description' => t('The default value of the field. For multiple selects use commas to separate multiple defaults.') . theme('webform_token_help'),
'#size' => 60,
'#maxlength' => 256,
'#weight' => 0,
);
$edit_fields['extra']['multiple'] = array(
'#type' => 'checkbox',
'#title' => t('Multiple'),
'#return_value' => 'Y',
'#default_value' => $currfield['extra']['multiple'],
'#description' => t('Check this option if the user should be allowed to choose multiple values.'),
);
$edit_fields['extra']['aslist'] = array(
'#type' => 'checkbox',
'#title' => t('Listbox'),
'#return_value' => 'Y',
'#default_value' => $currfield['extra']['aslist'],
'#description' => t('Check this option if you want the select component to be of listbox type instead of radiobuttons or checkboxes.'),
);
$edit_fields['extra']['email'] = array(
'#type' => 'checkbox',
'#title' => t('E-mail a submission copy'),
'#return_value' => 1,
'#default_value' => $currfield['extra']['email'],
'#description' => t('Check this option if this component contains an e-mail address that should get a copy of the submission. Emails are sent individually so other emails will not be shown to the recipient.') .' '.
t('To use the option with a select component, you must use key-value pairs seperated by pipes. i.e. user@example.com|Sample user.'),
);
return $edit_fields;
}
/**
* Element validation callback. Ensure keys are not duplicated.
*/
function _webform_edit_validate_select($element, $form_state) {
// TODO: Validate e-mail addresses when used as keys?\
// Check for duplicate key values to prevent unexpected data loss.
if (!empty($element['#value'])) {
$lines = explode("\n", $element['#value']);
$existing_keys = array();
$duplicate_keys = array();
$group = '';
foreach ($lines as $line) {
$matches = array();
$line = trim($line);
if (preg_match('/^\<([^>]*)\>$/', $line, $matches)) {
$group = $matches[1];
$key = NULL; // No need to store group names.
}
elseif (preg_match('/^([^|]+)\|(.*)$/', $line, $matches)) {
$key = $matches[1];
}
else {
$key = $line;
}
if (isset($key)) {
if (isset($existing_keys[$group][$key])) {
$duplicate_keys[$key] = $key;
}
else {
$existing_keys[$group][$key] = $key;
}
}
}
if (!empty($duplicate_keys)) {
form_error($element, t('Options within the select list must be unique. The following keys have been used multiple times:') . theme('item_list', $duplicate_keys));
}
}
return TRUE;
}
/**
* Build a form item array containing all the properties of this component.
* @param $component
* An array of information describing the component, directly correlating to
* the webform_component database schema.
* @return
* An array of a form item to be displayed on the client-side webform.
*/
function _webform_render_select($component) {
$form_item = array(
'#title' => $component['name'],
'#required' => $component['mandatory'],
'#weight' => $component['weight'],
'#description' => _webform_filter_descriptions($component['extra']['description']),
'#prefix' => '
',
'#suffix' => '
',
);
// Convert the user-entered options list into an array.
$default_value = _webform_filter_values($component['value'], NULL, NULL, FALSE);
$options = _webform_select_options($component['extra']['items'], $component['extra']['aslist'] !== 'Y');
if ($component['extra']['aslist'] === 'Y' && $component['extra']['multiple'] !== 'Y' && !($component['mandatory'] && !empty($component['value']))) {
$options = array('' => t('select...')) + $options;
}
// Set the component options.
$form_item['#options'] = $options;
// Set the default value.
if ($default_value != '') {
// Convert default value to a list if necessary.
if ($component['extra']['multiple'] === 'Y') {
$varray = array_filter(explode(',', $default_value));
foreach ($varray as $key => $v) {
$form_item['#default_value'][] = $v;
}
}
else {
$form_item['#default_value'] = $default_value;
}
}
if ($component['extra']['aslist'] === 'Y') {
// Set display as a select list:
$form_item['#type'] = 'select';
if ($component['extra']['multiple'] === 'Y') {
$form_item['#multiple'] = TRUE;
}
}
else {
if ($component['extra']['multiple'] === 'Y') {
// Set display as a checkbox set.
$form_item['#type'] = 'checkboxes';
// Drupal 6 hack to properly render on multipage forms.
$form_item['#process'] = array('webform_expand_checkboxes');
}
else {
// Set display as a radio set.
$form_item['#type'] = 'radios';
}
}
return $form_item;
}
/**
* Drupal 6 hack that properly *renders* checkboxes in multistep forms. This is
* different than the value hack needed in Drupal 5, which is no longer needed.
*/
function webform_expand_checkboxes($element) {
// Elements that have a value set are already in the form structure cause
// them not to be written when the expand_checkboxes function is called.
$default_value = array();
foreach (element_children($element) as $key) {
if (isset($element[$key]['#default_value'])) {
$default_value[$key] = $element[$key]['#default_value'];
unset($element[$key]);
}
}
$element = expand_checkboxes($element);
foreach ($default_value as $key=>$val) {
$element[$key]['#default_value'] = $val;
}
return $element;
}
/**
* Display the result of a textfield submission. The output of this function
* will be displayed under the "results" tab then "submissions".
* @param $data
* An array of information containing the submission result, directly
* correlating to the webform_submitted_data database schema.
* @param $component
* An array of information describing the component, directly correlating to
* the webform_component database schema.
* @param $enabled
* If enabled, the value may be changed. Otherwise it will set to readonly.
* @return
* Textual output formatted for human reading.
*/
function _webform_submission_display_select($data, $component, $enabled = FALSE) {
$form_item = _webform_render_select($component);
if ($component['extra']['multiple'] === 'Y') {
// Set the value as an array.
$form_item['#default_value'] = array();
foreach ((array)$data['value'] as $key => $value) {
$form_item['#default_value'][] = $value;
}
}
else {
// Set the value as a single string.
$form_item['#default_value'] = '';
foreach ((array)$data['value'] as $value) {
$form_item['#default_value'] = $value;
}
}
$form_item['#disabled'] = !$enabled;
return $form_item;
}
/**
* Convert FAPI 0/1 values into something saveable.
*
* @param $data
* The POST data associated with the component.
* @param $component
* An array of information describing the component, directly correlating to
* the webform_component database schema.
* @return
* Nothing.
*/
function _webform_submit_select(&$data, $component) {
$options = drupal_map_assoc(array_flip(_webform_select_options($component['extra']['items'], TRUE)));
if (is_array($data)) {
foreach ($data as $key => $value) {
if ($value != '') {
$data[$key] = $options[$key];
}
// Checkboxes submit a value of 0 when not checked.
elseif ($value == 0 && $component['extra']['aslist'] !== 'Y' && $component['extra']['multiple'] === 'Y') {
unset($data[$key]);
}
else {
unset($data[$key]);
}
}
}
}
/**
* Format the output of emailed data for this component.
*
* @param $data
* A string or array of the submitted data.
* @param $component
* An array of information describing the component, directly correlating to
* the webform_component database schema.
* @return
* Textual output to be included in the email.
*/
function theme_webform_mail_select($data, $component) {
// Convert submitted 'safe' values to un-edited, original form.
$options = _webform_select_options($component['extra']['items'], TRUE);
// Generate the output.
$output = '';
if ($component['extra']['multiple']) {
$output .= $component['name'] .":\n";
foreach ((array)$data as $value) {
if ($value) {
if ($options[$value]) {
$output .= ' - '. $options[$value] ."\n";
}
}
}
}
else {
if ($data !== '' && $options[$data]) {
$output .= $component['name'] .": ". $options[$data] ."\n";
}
}
return $output;
}
/**
* Module specific instance of hook_help().
*/
function _webform_help_select($section) {
switch ($section) {
case 'admin/settings/webform#select_description':
return t('Allows creation of checkboxes, radio buttons, or select menus.');
}
}
/**
* Module specific instance of hook_theme().
*/
function _webform_theme_select() {
return array(
'webform_mail_select' => array(
'arguments' => array('data' => NULL, 'component' => NULL),
),
);
}
/**
* Calculate and returns statistics about results for this component from all
* submission to this webform. The output of this function will be displayed
* under the "results" tab then "analysis".
* @param $component
* An array of information describing the component, directly correlating to
* the webform_component database schema
* @param $sids
* An optional array of submission IDs (sid). If supplied, the analysis will be limited
* to these sids.
* @return
* An array of data rows, each containing a statistic for this component's
* submissions.
*/
function _webform_analysis_rows_select($component, $sids = array()) {
$options = _webform_select_options($component['extra']['items'], TRUE);
$placeholders = count($sids) ? array_fill(0, count($sids), "'%s'") : array();
$sidfilter = count($sids) ? " AND sid in (".implode(",", $placeholders).")" : "";
$query = 'SELECT data, count(data) as datacount '.
' FROM {webform_submitted_data} '.
' WHERE nid = %d '.
' AND cid = %d '.
" AND data != '0' AND data != '' $sidfilter ".
' GROUP BY data ';
$result = db_query($query, array_merge(array($component['nid'], $component['cid']), $sids));
$rows = array();
while ($data = db_fetch_array($result)) {
if (isset($options[$data['data']])) {
$display_option = $options[$data['data']];
}
else {
$display_option = $data['data'];
}
$rows[] = array($display_option, $data['datacount']);
}
return $rows;
}
/**
* Return the result of this component's submission for display in a table. The
* output of this function will be displayed under the "results" tab then "table".
* @param $data
* An array of information containing the submission result, directly
* correlating to the webform_submitted_data database schema
* @return
* Textual output formatted for human reading.
*/
function _webform_table_data_select($data) {
$output = '';
// Set the value as a single string.
if (is_array($data['value'])) {
foreach ($data['value'] as $value) {
if ($value !== '0') {
$output .= check_plain($value) .'
';
}
}
}
else {
$output .= check_plain(empty($data['value']['0']) ? '' : $data['value']['0']);
}
return $output;
}
/**
* Return the header information for this component to be displayed in a comma
* seperated value file. The output of this function will be displayed under the
* "results" tab then "download".
* @param $component
* An array of information describing the component, directly correlating to
* the webform_component database schema.
* @return
* An array of data to be displayed in the first three rows of a CSV file, not
* including either prefixed or trailing commas.
*/
function _webform_csv_headers_select($component) {
$headers = array(
0 => array(),
1 => array(),
2 => array(),
);
if ($component['extra']['multiple']) {
$headers[0][] = '';
$headers[1][] = $component['name'];
$items = _webform_select_options($component['extra']['items'], TRUE);
$count = 0;
foreach ($items as $key => $item) {
// Empty column per sub-field in main header.
if ($count != 0) {
$headers[0][] = '';
$headers[1][] = '';
}
$headers[2][] = $key;
$count++;
}
}
else {
$headers[0][] = '';
$headers[1][] = '';
$headers[2][] = $component['name'];
}
return $headers;
}
/**
* Return the result of a textfield submission. The output of this function will
* be displayed under the "results" tab then "submissions".
* @param $data
* An array of information containing the submission result, directly
* correlating to the webform_submitted_data database schema.
* @return
* Textual output formatted for CSV, not including either prefixed or trailing
* commas.
*/
function _webform_csv_data_select($data, $component) {
$value = _webform_filter_values($component['value'], NULL, NULL, FALSE);
$options = _webform_select_options($component['extra']['items'], TRUE);
$return = array();
if ($component['extra']['multiple']) {
foreach ($options as $key => $item) {
if (in_array($key, (array)$data['value']) === TRUE) {
$return[] = 'X';
}
else {
$return[] = '';
}
}
}
else {
$return = $data['value'][0];
}
return $return;
}
/**
* Utility function to split user-entered values from new-line seperated
* text into an array of options.
*
* @param $text
* Text to be converted into a select option array.
* @param $flat
* Optional. If specified, return the option array and exclude any optgroups.
*/
function _webform_select_options($text, $flat = FALSE) {
$options = array();
$rows = array_filter(explode("\n", trim($text)));
$group = NULL;
foreach ($rows as $option) {
$option = trim($option);
/**
* If the Key of the option is within < >, treat as an optgroup
*
*
* creates an optgroup with the label "Group 1"
*
* <>
* Unsets the current group, allowing items to be inserted at the root element.
*/
if (preg_match('/^\<([^>]*)\>$/', $option, $matches)) {
if (empty($matches[1])) {
unset($group);
}
elseif (!$flat) {
$group = _webform_filter_values($matches[1], NULL, NULL, FALSE);
}
}
elseif (preg_match('/^([^|]+)\|(.*)$/', $option, $matches)) {
$key = _webform_filter_values($matches[1], NULL, NULL, FALSE);
$value = _webform_filter_values($matches[2], NULL, NULL, FALSE);
isset($group) ? $options[$group][$key] = $value : $options[$key] = $value;
}
else {
$filtered_option = _webform_filter_values($option, NULL, NULL, FALSE);
isset($group) ? $options[$group][$filtered_option] = $filtered_option : $options[$filtered_option] = $filtered_option;
}
}
return $options;
}