Newer
Older
/**
* @file
* Enables your site to capture votes on different topics in the form of multiple
* choice questions.
*/

Dries Buytaert
committed
/**

Dries Buytaert
committed
* Implements hook_help().

Dries Buytaert
committed
*/

Gábor Hojtsy
committed
function poll_help($path, $arg) {
switch ($path) {

Angie Byron
committed
$output = '';
$output .= '<h3>' . t('About') . '</h3>';

Jennifer Hodgdon
committed
$output .= '<p>' . t('The Poll module can be used to create simple surveys or questionnaires that display cumulative results. A poll is a good way to receive feedback from site users and community members. For more information, see the online handbook entry for the <a href="@poll">Poll module</a>.', array('@poll' => 'http://drupal.org/documentation/modules/poll/')) . '</p>';

Angie Byron
committed
$output .= '<h3>' . t('Uses') . '</h3>';
$output .= '<dl>';
$output .= '<dt>' . t('Creating a poll') . '</dt>';

Angie Byron
committed
$output .= '<dd>' . t('Users can create a poll by clicking on Poll on the <a href="@add-content">Add new content</a> page, and entering the question being posed, the answer choices, and beginning vote counts for each choice. The status (closed or active) and duration (length of time the poll remains active for new votes) can also be specified.', array('@add-content' => url('node/add'))) . '</dd>';

Angie Byron
committed
$output .= '<dt>' . t('Viewing polls') . '</dt>';
$output .= '<dd>' . t('You can visit the <a href="@poll">Polls</a> page to view all current polls, or alternately enable the <em>Most recent poll</em> block on the <a href="@blocks">Blocks administration page</a>. To vote in or view the results of a specific poll, you can click on the poll itself.', array('@poll' => url('poll'), '@blocks' => url('admin/structure/block'))) . '</dd>';

Angie Byron
committed
$output .= '</dl>';

Dries Buytaert
committed
return $output;

Dries Buytaert
committed
/**

Dries Buytaert
committed
* Implements hook_theme().

Dries Buytaert
committed
*/
function poll_theme() {

Angie Byron
committed
$theme_hooks = array(

Dries Buytaert
committed
'poll_vote' => array(
'template' => 'poll-vote',
'render element' => 'form',

Dries Buytaert
committed
),
'poll_choices' => array(
'render element' => 'form',
),

Dries Buytaert
committed
'poll_results' => array(
'template' => 'poll-results',
'variables' => array('raw_title' => NULL, 'results' => NULL, 'votes' => NULL, 'raw_links' => NULL, 'block' => NULL, 'nid' => NULL, 'vote' => NULL),

Dries Buytaert
committed
),
'poll_bar' => array(
'template' => 'poll-bar',
'variables' => array('title' => NULL, 'votes' => NULL, 'total_votes' => NULL, 'vote' => NULL, 'block' => NULL),

Dries Buytaert
committed
),
);

Angie Byron
committed
// The theme system automatically discovers the theme's functions and
// templates that implement more targeted "suggestions" of generic theme
// hooks. But suggestions implemented by a module must be explicitly
// registered.
$theme_hooks += array(
'poll_results__block' => array(
'template' => 'poll-results--block',
'variables' => $theme_hooks['poll_results']['variables'],
),
'poll_bar__block' => array(
'template' => 'poll-bar--block',
'variables' => $theme_hooks['poll_bar']['variables'],
),
);
return $theme_hooks;

Dries Buytaert
committed
}

Dries Buytaert
committed
/**

Dries Buytaert
committed
* Implements hook_permission().

Dries Buytaert
committed
*/
function poll_permission() {

Dries Buytaert
committed
$perms = array(

Angie Byron
committed
'vote on polls' => array(
'title' => t('Vote on polls'),
),
'cancel own vote' => array(
'title' => t('Cancel and change own votes'),

Angie Byron
committed
),
'inspect all votes' => array(
'title' => t('View details for all votes'),

Angie Byron
committed
),
);
return $perms;

Dries Buytaert
committed
}
/**

Dries Buytaert
committed
* Implements hook_menu().

Dries Buytaert
committed
*/
function poll_menu() {
$items['poll'] = array(
'title' => 'Polls',
'page callback' => 'poll_page',
'access arguments' => array('access content'),
'type' => MENU_SUGGESTED_ITEM,
'file' => 'poll.pages.inc',

Dries Buytaert
committed
);
$items['node/%node/votes'] = array(
'title' => 'Votes',
'page callback' => 'poll_votes',
'page arguments' => array(1),
'access callback' => '_poll_menu_access',
'access arguments' => array(1, 'inspect all votes', FALSE),
'weight' => 3,
'type' => MENU_LOCAL_TASK,
'file' => 'poll.pages.inc',

Dries Buytaert
committed
);
$items['node/%node/results'] = array(
'title' => 'Results',
'page callback' => 'poll_results',
'page arguments' => array(1),
'access callback' => '_poll_menu_access',
'access arguments' => array(1, 'access content', TRUE),
'weight' => 3,
'type' => MENU_LOCAL_TASK,
'file' => 'poll.pages.inc',

Dries Buytaert
committed
);

Dries Buytaert
committed
return $items;
}
/**
* Callback function to see if a node is acceptable for poll menu items.
*/
function _poll_menu_access($node, $perm, $inspect_allowvotes) {

Dries Buytaert
committed
return user_access($perm) && ($node->type == 'poll') && ($node->allowvotes || !$inspect_allowvotes);
}

Dries Buytaert
committed
/**

Dries Buytaert
committed
* Implements hook_block_info().

Dries Buytaert
committed
*/

Dries Buytaert
committed
function poll_block_info() {

Dries Buytaert
committed
$blocks['recent']['info'] = t('Most recent poll');

Angie Byron
committed
$blocks['recent']['properties']['administrative'] = TRUE;

Dries Buytaert
committed
return $blocks;

Dries Buytaert
committed
}
/**

Dries Buytaert
committed
* Implements hook_block_view().

Dries Buytaert
committed
*
* Generates a block containing the latest poll.
*/

Dries Buytaert
committed
function poll_block_view($delta = '') {

Dries Buytaert
committed
// Retrieve the latest poll.

Dries Buytaert
committed
$select = db_select('node', 'n');
$select->join('poll', 'p', 'p.nid = n.nid');
$select->fields('n', array('nid'))
->condition('n.status', 1)
->condition('p.active', 1)
->orderBy('n.created', 'DESC')
->range(0, 1)
->addTag('node_access');
$record = $select->execute()->fetchObject();

Dries Buytaert
committed
if ($record) {
$poll = node_load($record->nid);
if ($poll->nid) {
$poll = poll_block_latest_poll_view($poll);

Dries Buytaert
committed
$block['subject'] = t('Poll');
$block['content'] = $poll->content;

Dries Buytaert
committed
return $block;

Dries Buytaert
committed
/**

Dries Buytaert
committed
* Implements hook_cron().

Dries Buytaert
committed
*
* Closes polls that have exceeded their allowed runtime.
*/

Dries Buytaert
committed
$nids = db_query('SELECT p.nid FROM {poll} p INNER JOIN {node} n ON p.nid = n.nid WHERE (n.created + p.runtime) < :request_time AND p.active = :active AND p.runtime <> :runtime', array(':request_time' => REQUEST_TIME, ':active' => 1, ':runtime' => 0))->fetchCol();
if (!empty($nids)) {
db_update('poll')
->fields(array('active' => 0))
->condition('nid', $nids, 'IN')
->execute();

Dries Buytaert
committed
/**

Dries Buytaert
committed
* Implements hook_node_info().

Dries Buytaert
committed
*/

Dries Buytaert
committed
function poll_node_info() {
return array(
'poll' => array(
'name' => t('Poll'),
'base' => 'poll',
'description' => t('A <em>poll</em> is a question with a set of possible responses. A <em>poll</em>, once created, automatically provides a simple running count of the number of votes received for each response.'),

Dries Buytaert
committed
'title_label' => t('Question'),
'has_body' => FALSE,
)
);

Dries Buytaert
committed
}

Angie Byron
committed
/**

Dries Buytaert
committed
* Implements hook_field_extra_fields().

Angie Byron
committed
*/
function poll_field_extra_fields() {
$extra['node']['poll'] = array(

Dries Buytaert
committed
'form' => array(
'choice_wrapper' => array(
'label' => t('Poll choices'),
'description' => t('Poll choices'),
'weight' => -4,
),
'settings' => array(
'label' => t('Poll settings'),
'description' => t('Poll module settings'),
'weight' => -3,
),

Dries Buytaert
committed
'display' => array(
'poll_view_voting' => array(
'label' => t('Poll vote'),
'description' => t('Poll vote'),
'weight' => 0,
),
'poll_view_results' => array(
'label' => t('Poll results'),
'description' => t('Poll results'),
'weight' => 0,
),
)

Angie Byron
committed
return $extra;
}

Dries Buytaert
committed
/**

Dries Buytaert
committed
* Implements hook_form().

Dries Buytaert
committed
*/
function poll_form($node, &$form_state) {
global $user;

Dries Buytaert
committed
$admin = user_access('bypass node access') || user_access('edit any poll content') || (user_access('edit own poll content') && $user->uid == $node->uid);

Angie Byron
committed
$type = node_type_get_type($node);

Dries Buytaert
committed
// The submit handlers to add more poll choices require that this form is
// cached, regardless of whether Ajax is used.
$form_state['cache'] = TRUE;
$form['title'] = array(
'#type' => 'textfield',
'#title' => check_plain($type->title_label),
'#required' => TRUE,
'#default_value' => $node->title,
'#weight' => -5,
);
if (isset($form_state['choice_count'])) {
$choice_count = $form_state['choice_count'];
}
else {
$choice_count = max(2, empty($node->choice) ? 2 : count($node->choice));
// Add a wrapper for the choices and more button.
$form['choice_wrapper'] = array(
'#tree' => FALSE,

Gábor Hojtsy
committed
'#weight' => -4,

Angie Byron
committed
'#prefix' => '<div class="clearfix" id="poll-choice-wrapper">',
'#suffix' => '</div>',
);
// Container for just the poll choices.
$form['choice_wrapper']['choice'] = array(
'#prefix' => '<div id="poll-choices">',
'#suffix' => '</div>',
'#theme' => 'poll_choices',
);
// Add the current choices to the form.

Dries Buytaert
committed
$delta = 0;
$weight = 0;
if (isset($node->choice)) {
$delta = count($node->choice);
foreach ($node->choice as $chid => $choice) {
$key = 'chid:' . $chid;

Dries Buytaert
committed
$form['choice_wrapper']['choice'][$key] = _poll_choice_form($key, $choice['chid'], $choice['chtext'], $choice['chvotes'], $choice['weight'], $choice_count);
$weight = max($choice['weight'], $weight);

Dries Buytaert
committed
}
}

Dries Buytaert
committed
// Add initial or additional choices.
$existing_delta = $delta;
for ($delta; $delta < $choice_count; $delta++) {
$key = 'new:' . ($delta - $existing_delta);

Angie Byron
committed
// Increase the weight of each new choice.
$weight++;

Dries Buytaert
committed
$form['choice_wrapper']['choice'][$key] = _poll_choice_form($key, NULL, '', 0, $weight, $choice_count);
}
// We name our button 'poll_more' to avoid conflicts with other modules using
// Ajax-enabled buttons with the id 'more'.
$form['choice_wrapper']['poll_more'] = array(
'#type' => 'submit',
'#value' => t('More choices'),

Dries Buytaert
committed
'#attributes' => array(
'title' => t("If the amount of boxes above isn't enough, click here to add more choices."),
),
'#weight' => 1,

Angie Byron
committed
'#limit_validation_errors' => array(array('choice')),
'#submit' => array('poll_more_choices_submit'),

Angie Byron
committed
'#ajax' => array(

Angie Byron
committed
'callback' => 'poll_choice_js',
'wrapper' => 'poll-choices',
'effect' => 'fade',
),
);

Dries Buytaert
committed
$duration = array(
// 1-6 days.
86400, 2 * 86400, 3 * 86400, 4 * 86400, 5 * 86400, 6 * 86400,
// 1-3 weeks (7 days).
604800, 2 * 604800, 3 * 604800,
// 1-3,6,9 months (30 days).
2592000, 2 * 2592000, 3 * 2592000, 6 * 2592000, 9 * 2592000,
// 1 year (365 days).
31536000,
);
$duration = array(0 => t('Unlimited')) + drupal_map_assoc($duration, 'format_interval');

Dries Buytaert
committed
$active = array(0 => t('Closed'), 1 => t('Active'));
$form['settings'] = array(
'#type' => 'fieldset',
'#collapsible' => TRUE,
'#title' => t('Poll settings'),
'#weight' => -3,
'#access' => $admin,
);

Dries Buytaert
committed
$form['settings']['active'] = array(
'#type' => 'radios',
'#title' => t('Poll status'),
'#default_value' => isset($node->active) ? $node->active : 1,
'#options' => $active,
'#description' => t('When a poll is closed, visitors can no longer vote for it.'),
'#access' => $admin,
);
$form['settings']['runtime'] = array(
'#type' => 'select',
'#title' => t('Poll duration'),

Steven Wittens
committed
'#default_value' => isset($node->runtime) ? $node->runtime : 0,

Dries Buytaert
committed
'#options' => $duration,
'#description' => t('After this period, the poll will be closed automatically.'),
);
/**

Angie Byron
committed
* Submit handler to add more choices to a poll form.
*
* This handler is run regardless of whether JS is enabled or not. It makes
* changes to the form state. If the button was clicked with JS disabled, then
* the page is reloaded with the complete rebuilt form. If the button was
* clicked with JS enabled, then ajax_form_callback() calls poll_choice_js() to
* return just the changed part of the form.
*/
function poll_more_choices_submit($form, &$form_state) {
// If this is a Ajax POST, add 1, otherwise add 5 more choices to the form.
if ($form_state['values']['poll_more']) {

Angie Byron
committed
$n = $_GET['q'] == 'system/ajax' ? 1 : 5;
$form_state['choice_count'] = count($form_state['values']['choice']) + $n;
}

Dries Buytaert
committed
// Renumber the choices. This invalidates the corresponding key/value
// associations in $form_state['input'], so clear that out. This requires
// poll_form() to rebuild the choices with the values in
// $form_state['node']->choice, which it does.
$form_state['node']->choice = array_values($form_state['values']['choice']);
unset($form_state['input']['choice']);
$form_state['rebuild'] = TRUE;
}

Dries Buytaert
committed
function _poll_choice_form($key, $chid = NULL, $value = '', $votes = 0, $weight = 0, $size = 10) {
$form = array(
'#tree' => TRUE,
'#weight' => $weight,
);
// We'll manually set the #parents property of these fields so that
// their values appear in the $form_state['values']['choice'] array.

Dries Buytaert
committed
$form['chid'] = array(
'#type' => 'value',
'#value' => $chid,
'#parents' => array('choice', $key, 'chid'),
);
$form['chtext'] = array(
'#type' => 'textfield',

Dries Buytaert
committed
'#title' => $value !== '' ? t('Choice label') : t('New choice label'),
'#title_display' => 'invisible',
'#default_value' => $value,

Dries Buytaert
committed
'#parents' => array('choice', $key, 'chtext'),
);

Dries Buytaert
committed
$form['chvotes'] = array(
'#type' => 'textfield',

Dries Buytaert
committed
'#title' => $value !== '' ? t('Vote count for choice @label', array('@label' => $value)) : t('Vote count for new choice'),
'#title_display' => 'invisible',

Dries Buytaert
committed
'#default_value' => $votes,
'#size' => 5,
'#maxlength' => 7,
'#parents' => array('choice', $key, 'chvotes'),
'#access' => user_access('administer nodes'),

Angie Byron
committed
'#element_validate' => array('element_validate_integer'),

Dries Buytaert
committed
);
$form['weight'] = array(
'#type' => 'weight',

Dries Buytaert
committed
'#title' => $value !== '' ? t('Weight for choice @label', array('@label' => $value)) : t('Weight for new choice'),
'#title_display' => 'invisible',

Dries Buytaert
committed
'#default_value' => $weight,
'#delta' => $size,
'#parents' => array('choice', $key, 'weight'),
);
return $form;
}
/**

Angie Byron
committed
* Ajax callback in response to new choices being added to the form.
*
* This returns the new page content to replace the page content made obsolete
* by the form submission.
*
* @see poll_more_choices_submit()
*/

Angie Byron
committed
function poll_choice_js($form, $form_state) {

Dries Buytaert
committed
return $form['choice_wrapper']['choice'];
}

Dries Buytaert
committed
/**

Dries Buytaert
committed
* Form submit handler for node_form().
*
* Upon preview and final submission, we need to renumber poll choices and
* create a teaser output.

Dries Buytaert
committed
*/
function poll_node_form_submit(&$form, &$form_state) {

Dries Buytaert
committed
// Renumber choices.

Dries Buytaert
committed
$form_state['values']['choice'] = array_values($form_state['values']['choice']);
$form_state['values']['teaser'] = poll_teaser((object) $form_state['values']);

Dries Buytaert
committed
* Implements hook_validate().
function poll_validate($node, $form) {

Dries Buytaert
committed
if (isset($node->title)) {

Angie Byron
committed
// Check for at least two options and validate amount of votes.

Dries Buytaert
committed
$realchoices = 0;
foreach ($node->choice as $i => $choice) {
if ($choice['chtext'] != '') {
$realchoices++;
}
if (isset($choice['chvotes']) && $choice['chvotes'] < 0) {

Dries Buytaert
committed
form_set_error("choice][$i][chvotes", t('Negative values are not allowed.'));
}
}

Dries Buytaert
committed
if ($realchoices < 2) {
form_set_error("choice][$realchoices][chtext", t('You must fill in at least two choices.'));
}
}

Dries Buytaert
committed
}

Angie Byron
committed
* Implements hook_field_attach_prepare_translation_alter().

Angie Byron
committed
function poll_field_attach_prepare_translation_alter(&$entity, $context) {
if ($context['entity_type'] == 'node' && $entity->type == 'poll') {
$entity->choice = $context['source_entity']->choice;

Angie Byron
committed
foreach ($entity->choice as $i => $options) {
$entity->choice[$i]['chvotes'] = 0;
}
}
}

Dries Buytaert
committed
/**

Dries Buytaert
committed
* Implements hook_load().

Dries Buytaert
committed
*/
function poll_load($nodes) {
global $user;
foreach ($nodes as $node) {

Dries Buytaert
committed
$poll = db_query("SELECT runtime, active FROM {poll} WHERE nid = :nid", array(':nid' => $node->nid))->fetchObject();
if (empty($poll)) {
$poll = new stdClass();
// Load the appropriate choices into the $poll object.

Dries Buytaert
committed
$poll->choice = db_select('poll_choice', 'c')
->addTag('translatable')
->fields('c', array('chid', 'chtext', 'chvotes', 'weight'))
->condition('c.nid', $node->nid)

Dries Buytaert
committed
->orderBy('weight')
->execute()->fetchAllAssoc('chid', PDO::FETCH_ASSOC);
// Determine whether or not this user is allowed to vote.
$poll->allowvotes = FALSE;
if (user_access('vote on polls') && $poll->active) {
if ($user->uid) {

Dries Buytaert
committed
// If authenticated, find existing vote based on uid.

Dries Buytaert
committed
$poll->vote = db_query('SELECT chid FROM {poll_vote} WHERE nid = :nid AND uid = :uid', array(':nid' => $node->nid, ':uid' => $user->uid))->fetchField();
if (empty($poll->vote)) {
$poll->vote = -1;
$poll->allowvotes = TRUE;
}
}

Dries Buytaert
committed
elseif (!empty($_SESSION['poll_vote'][$node->nid])) {

Dries Buytaert
committed
// Otherwise the user is anonymous. Look for an existing vote in the
// user's session.

Dries Buytaert
committed
$poll->vote = $_SESSION['poll_vote'][$node->nid];
}
else {

Dries Buytaert
committed
// Finally, query the database for an existing vote based on anonymous
// user's hostname.
$poll->allowvotes = !db_query("SELECT 1 FROM {poll_vote} WHERE nid = :nid AND hostname = :hostname AND uid = 0", array(':nid' => $node->nid, ':hostname' => ip_address()))->fetchField();
}
foreach ($poll as $key => $value) {
$nodes[$node->nid]->$key = $value;
}
}

Dries Buytaert
committed
/**

Dries Buytaert
committed
* Implements hook_insert().

Dries Buytaert
committed
*/
function poll_insert($node) {

Dries Buytaert
committed
if (!user_access('administer nodes')) {
// Make sure all votes are 0 initially
foreach ($node->choice as $i => $choice) {
$node->choice[$i]['chvotes'] = 0;
}
$node->active = 1;
}

Dries Buytaert
committed
db_insert('poll')
->fields(array(
'nid' => $node->nid,
'runtime' => $node->runtime,
'active' => $node->active,
))
->execute();

Dries Buytaert
committed
foreach ($node->choice as $choice) {
if ($choice['chtext'] != '') {

Dries Buytaert
committed
db_insert('poll_choice')
->fields(array(
'nid' => $node->nid,
'chtext' => $choice['chtext'],
'chvotes' => $choice['chvotes'],
'weight' => $choice['weight'],
))
->execute();

Dries Buytaert
committed
}
}
}
/**

Dries Buytaert
committed
* Implements hook_update().

Dries Buytaert
committed
*/
function poll_update($node) {

Gábor Hojtsy
committed
// Update poll settings.

Dries Buytaert
committed
db_update('poll')
->fields(array(
'runtime' => $node->runtime,
'active' => $node->active,
))
->condition('nid', $node->nid)
->execute();

Dries Buytaert
committed

Dries Buytaert
committed
// Poll choices with empty titles signifies removal. We remove all votes to
// the removed options, so people who voted on them can vote again.
foreach ($node->choice as $key => $choice) {
if (!empty($choice['chtext'])) {

Dries Buytaert
committed
db_merge('poll_choice')
->key(array('chid' => $choice['chid']))
->fields(array(
'chtext' => $choice['chtext'],
'chvotes' => (int) $choice['chvotes'],
'weight' => $choice['weight'],
))
->insertFields(array(

Angie Byron
committed
'nid' => $node->nid,
'chtext' => $choice['chtext'],
'chvotes' => (int) $choice['chvotes'],
'weight' => $choice['weight'],
))

Dries Buytaert
committed
->execute();

Gábor Hojtsy
committed
}
else {

Dries Buytaert
committed
db_delete('poll_vote')
->condition('nid', $node->nid)
->condition('chid', $key)
->execute();

Angie Byron
committed
db_delete('poll_choice')
->condition('nid', $node->nid)
->condition('chid', $choice['chid'])
->execute();

Dries Buytaert
committed
}
}
}
/**

Dries Buytaert
committed
* Implements hook_delete().

Dries Buytaert
committed
*/
function poll_delete($node) {

Dries Buytaert
committed
db_delete('poll')
->condition('nid', $node->nid)
->execute();
db_delete('poll_choice')
->condition('nid', $node->nid)
->execute();
db_delete('poll_vote')
->condition('nid', $node->nid)
->execute();

Dries Buytaert
committed
}
/**
* Return content for 'latest poll' block.

Dries Buytaert
committed
*
* @param $node
* The node object to load.

Dries Buytaert
committed
*/
function poll_block_latest_poll_view($node) {

Dries Buytaert
committed
global $user;
$output = '';
// This is necessary for shared objects because PHP doesn't copy objects, but
// passes them by reference. So when the objects are cached it can result in
// the wrong output being displayed on subsequent calls. The cloning and
// unsetting of $node->content prevents the block output from being the same
// as the node output.
$node = clone $node;
unset($node->content);
// No 'read more' link.
$node->readmore = FALSE;
$node->teaser = '';

Dries Buytaert
committed
$links = array();
$links[] = array('title' => t('Older polls'), 'href' => 'poll', 'attributes' => array('title' => t('View the list of polls on this site.')));
if ($node->allowvotes) {
$links[] = array('title' => t('Results'), 'href' => 'node/' . $node->nid . '/results', 'attributes' => array('title' => t('View the current poll results.')));
}

Dries Buytaert
committed
$node->links = $links;

Dries Buytaert
committed
if (!empty($node->allowvotes)) {
$node->content['poll_view_voting'] = drupal_get_form('poll_view_voting', $node, TRUE);
$node->content['links'] = array(
'#theme' => 'links',
'#links' => $node->links,
'#weight' => 5,
);
}
else {
$node->content['poll_view_results'] = array('#markup' => poll_view_results($node, TRUE, TRUE));

Dries Buytaert
committed
}
return $node;
}
/**

Dries Buytaert
committed
* Implements hook_view().

Dries Buytaert
committed
function poll_view($node, $view_mode) {
global $user;
$output = '';
if (!empty($node->allowvotes) && empty($node->show_results)) {
$node->content['poll_view_voting'] = drupal_get_form('poll_view_voting', $node);

Dries Buytaert
committed
}
else {
$node->content['poll_view_results'] = array('#markup' => poll_view_results($node, $view_mode));

Dries Buytaert
committed
}
return $node;

Dries Buytaert
committed
/**

Dries Buytaert
committed
* Creates a simple teaser that lists all the choices.

Dries Buytaert
committed
*
* This is primarily used for RSS.

Dries Buytaert
committed
*/
function poll_teaser($node) {
$teaser = NULL;
if ($choice['chtext'] != '') {
$teaser .= '* ' . check_plain($choice['chtext']) . "\n";
}
}
return $teaser;
}

Dries Buytaert
committed
* Generates the voting form for a poll.

Dries Buytaert
committed
*
* @ingroup forms
* @see poll_vote()
* @see phptemplate_preprocess_poll_vote()
function poll_view_voting($form, &$form_state, $node, $block = FALSE) {
$list = array();
foreach ($node->choice as $i => $choice) {
$list[$i] = check_plain($choice['chtext']);
$form['choice'] = array(
'#type' => 'radios',

Dries Buytaert
committed
'#title' => t('Choices'),
'#title_display' => 'invisible',
'#default_value' => -1,
'#options' => $list,
);

Dries Buytaert
committed

Dries Buytaert
committed
$form['vote'] = array(

Dries Buytaert
committed
'#type' => 'submit',

Dries Buytaert
committed
'#value' => t('Vote'),
'#submit' => array('poll_vote'),
);
// Store the node so we can get to it in submit functions.
$form['#node'] = $node;
$form['#block'] = $block;
// Set form caching because we could have multiple of these forms on
// the same page, and we want to ensure the right one gets picked.
$form_state['cache'] = TRUE;

Dries Buytaert
committed
// Provide a more cleanly named voting form theme.
$form['#theme'] = 'poll_vote';

Dries Buytaert
committed
return $form;
/**
* Validation function for processing votes
*/
function poll_view_voting_validate($form, &$form_state) {
if ($form_state['values']['choice'] == -1) {
form_set_error( 'choice', t('Your vote could not be recorded because you did not select any of the choices.'));
}
}

Dries Buytaert
committed
/**
* Submit handler for processing a vote.

Dries Buytaert
committed
*/
function poll_vote($form, &$form_state) {
$node = $form['#node'];
$choice = $form_state['values']['choice'];
global $user;

Dries Buytaert
committed
db_insert('poll_vote')
->fields(array(
'nid' => $node->nid,
'chid' => $choice,
'uid' => $user->uid,

Dries Buytaert
committed
'hostname' => ip_address(),

Dries Buytaert
committed
'timestamp' => REQUEST_TIME,

Dries Buytaert
committed
))
->execute();

Dries Buytaert
committed
// Add one to the votes.

Dries Buytaert
committed
db_update('poll_choice')
->expression('chvotes', 'chvotes + 1')
->condition('chid', $choice)
->execute();

Dries Buytaert
committed
cache_clear_all();

Dries Buytaert
committed
if (!$user->uid) {
// The vote is recorded so the user gets the result view instead of the
// voting form when viewing the poll. Saving a value in $_SESSION has the
// convenient side effect of preventing the user from hitting the page
// cache. When anonymous voting is allowed, the page cache should only
// contain the voting form, not the results.
$_SESSION['poll_vote'][$node->nid] = $choice;
}

Dries Buytaert
committed
drupal_set_message(t('Your vote was recorded.'));
// Return the user to whatever page they voted from.
}
/**
* Themes the voting form for a poll.

Dries Buytaert
committed
*
* Inputs: $form

Dries Buytaert
committed
function template_preprocess_poll_vote(&$variables) {
$form = $variables['form'];
$variables['choice'] = drupal_render($form['choice']);
$variables['title'] = check_plain($form['#node']->title);

Dries Buytaert
committed
$variables['vote'] = drupal_render($form['vote']);
$variables['rest'] = drupal_render_children($form);

Dries Buytaert
committed
$variables['block'] = $form['#block'];
if ($variables['block']) {

Angie Byron
committed
$variables['theme_hook_suggestions'][] = 'poll_vote__block';

Dries Buytaert
committed
/**
* Generates a graphical representation of the results of a poll.
*/
function poll_view_results($node, $view_mode, $block = FALSE) {
// Make sure that choices are ordered by their weight.
uasort($node->choice, 'drupal_sort_weight');
// Count the votes and find the maximum.

Steven Wittens
committed
$total_votes = 0;
$max_votes = 0;
if (isset($choice['chvotes'])) {
$total_votes += $choice['chvotes'];
$max_votes = max($max_votes, $choice['chvotes']);
}

Steven Wittens
committed
$poll_results = '';
if (!empty($choice['chtext'])) {

Gábor Hojtsy
committed
$chvotes = isset($choice['chvotes']) ? $choice['chvotes'] : NULL;

Dries Buytaert
committed
$poll_results .= theme('poll_bar', array('title' => $choice['chtext'], 'votes' => $chvotes, 'total_votes' => $total_votes, 'vote' => isset($node->vote) && $node->vote == $i, 'block' => $block));

Dries Buytaert
committed
}
return theme('poll_results', array('raw_title' => $node->title, 'results' => $poll_results, 'votes' => $total_votes, 'raw_links' => isset($node->links) ? $node->links : array(), 'block' => $block, 'nid' => $node->nid, 'vote' => isset($node->vote) ? $node->vote : NULL));
/**
* Returns HTML for an admin poll form for choices.
*
* @param $variables
* An associative array containing:
* - form: A render element representing the form.

Gábor Hojtsy
committed
*
* @ingroup themeable
*/

Dries Buytaert
committed
function theme_poll_choices($variables) {
$form = $variables['form'];

Dries Buytaert
committed
drupal_add_tabledrag('poll-choice-table', 'order', 'sibling', 'poll-weight');

Gábor Hojtsy
committed
$is_admin= user_access('administer nodes');

Dries Buytaert
committed
$delta = 0;
$rows = array();
$headers = array('', t('Choice'));
if ($is_admin) {
$headers[] = t('Vote count');
}
$headers[] = t('Weight');
foreach (element_children($form) as $key) {

Dries Buytaert
committed
$delta++;
// Set special classes for drag and drop updating.
$form[$key]['weight']['#attributes']['class'] = array('poll-weight');
// Build the table row.
$row = array(
'data' => array(
array('class' => array('choice-flag')),

Dries Buytaert
committed
drupal_render($form[$key]['chtext']),
),
'class' => array('draggable'),
);
if ($is_admin) {
$row['data'][] = drupal_render($form[$key]['chvotes']);
}
$row['data'][] = drupal_render($form[$key]['weight']);

Dries Buytaert
committed
// Add any additional classes set on the row.
if (!empty($form[$key]['#attributes']['class'])) {
$row['class'] = array_merge($row['class'], $form[$key]['#attributes']['class']);
}

Dries Buytaert
committed
$rows[] = $row;
}

Dries Buytaert
committed
$output = theme('table', array('header' => $headers, 'rows' => $rows, 'attributes' => array('id' => 'poll-choice-table')));
$output .= drupal_render_children($form);
return $output;
}

Dries Buytaert
committed
* Preprocess the poll_results theme hook.
*
* Inputs: $raw_title, $results, $votes, $raw_links, $block, $nid, $vote. The
* $raw_* inputs to this are naturally unsafe; often safe versions are
* made to simply overwrite the raw version, but in this case it seems likely
* that the title and the links may be overridden by the theme layer, so they
* are left in with a different name for that purpose.
*
* @see poll-results.tpl.php
* @see poll-results--block.tpl.php

Dries Buytaert
committed
function template_preprocess_poll_results(&$variables) {

Angie Byron
committed
$variables['links'] = theme('links__poll_results', array('links' => $variables['raw_links']));

Dries Buytaert
committed
if (isset($variables['vote']) && $variables['vote'] > -1 && user_access('cancel own vote')) {

Dries Buytaert
committed
$elements = drupal_get_form('poll_cancel_form', $variables['nid']);
$variables['cancel_form'] = drupal_render($elements);

Dries Buytaert
committed
$variables['title'] = check_plain($variables['raw_title']);
if ($variables['block']) {

Angie Byron
committed
$variables['theme_hook_suggestions'][] = 'poll_results__block';

Dries Buytaert
committed
/**

Dries Buytaert
committed
* Preprocess the poll_bar theme hook.
*
* Inputs: $title, $votes, $total_votes, $voted, $block
*
* @see poll-bar.tpl.php
* @see poll-bar--block.tpl.php

Dries Buytaert
committed
*/

Dries Buytaert
committed
function template_preprocess_poll_bar(&$variables) {
if ($variables['block']) {

Angie Byron
committed
$variables['theme_hook_suggestions'][] = 'poll_bar__block';

Dries Buytaert
committed
}

Dries Buytaert
committed
$variables['title'] = check_plain($variables['title']);
$variables['percentage'] = round($variables['votes'] * 100 / max($variables['total_votes'], 1));

Dries Buytaert
committed
}

Dries Buytaert
committed
* Builds the cancel form for a poll.
*
* @ingroup forms
* @see poll_cancel()
function poll_cancel_form($form, &$form_state, $nid) {
$form_state['cache'] = TRUE;

Dries Buytaert
committed
// Store the nid so we can get to it in submit functions.
$form['#nid'] = $nid;

Dries Buytaert
committed
$form['actions'] = array('#type' => 'actions');
$form['actions']['submit'] = array(

Dries Buytaert
committed
'#type' => 'submit',

Dries Buytaert
committed
'#value' => t('Cancel your vote'),
'#submit' => array('poll_cancel')
);

Dries Buytaert
committed
return $form;
}

Dries Buytaert
committed
/**
* Submit callback for poll_cancel_form().

Dries Buytaert
committed
*/

Dries Buytaert
committed
function poll_cancel($form, &$form_state) {

Dries Buytaert
committed
global $user;

Dries Buytaert
committed
$node = node_load($form['#nid']);

Dries Buytaert
committed

Dries Buytaert
committed
db_delete('poll_vote')
->condition('nid', $node->nid)
->condition($user->uid ? 'uid' : 'hostname', $user->uid ? $user->uid : ip_address())
->execute();

Dries Buytaert
committed
// Subtract from the votes.

Dries Buytaert
committed
db_update('poll_choice')
->expression('chvotes', 'chvotes - 1')
->condition('chid', $node->vote)
->execute();

Dries Buytaert
committed
unset($_SESSION['poll_vote'][$node->nid]);
drupal_set_message(t('Your vote was cancelled.'));

Gerhard Killesreiter
committed
/**

Dries Buytaert
committed
* Implements hook_user_cancel().

Gerhard Killesreiter
committed
*/
function poll_user_cancel($edit, $account, $method) {

Angie Byron
committed
switch ($method) {
case 'user_cancel_reassign':

Dries Buytaert
committed
db_update('poll_vote')
->fields(array('uid' => 0))
->condition('uid', $account->uid)
->execute();

Angie Byron
committed
break;
}

Gerhard Killesreiter
committed
}

Dries Buytaert
committed

Dries Buytaert
committed
/**
* Implements hook_user_delete().
*/
function poll_user_delete($account) {
db_delete('poll_vote')