'admin/build/flags',
'title' => t('Flags'),
'callback' => 'flag_admin_page',
'access' => user_access('administer flags'),
'description' => t('Configure flags for marking content with arbitary information (such as offensive or bookmarked).'),
'type' => MENU_NORMAL_ITEM,
);
$items[] = array(
'path' => 'admin/build/flags/edit',
'title' => t('Edit flags'),
'callback' => 'drupal_get_form',
'callback arguments' => array('flag_form'),
'access' => user_access('administer flags'),
'type' => MENU_CALLBACK,
);
$items[] = array(
'path' => 'admin/build/flags/delete',
'title' => t('Delete flags'),
'callback' => 'drupal_get_form',
'callback arguments' => array('flag_delete_confirm'),
'access' => user_access('administer flags'),
'type' => MENU_CALLBACK,
);
$items[] = array(
'path' => 'admin/build/flags/list',
'title' => t('List'),
'callback' => 'flag_admin_page',
'access' => user_access('administer flags'),
'type' => MENU_DEFAULT_LOCAL_TASK,
'weight' => -1,
);
$items[] = array(
'path' => 'admin/build/flags/add',
'title' => t('Add'),
'callback' => 'flag_add_page',
'access' => user_access('administer flags'),
'type' => MENU_LOCAL_TASK,
);
$items[] = array(
'path' => 'flag',
'title' => t('Flag'),
'callback' => 'flag_page',
'access' => user_access('access content'),
'type' => MENU_CALLBACK,
);
$items[] = array(
'path' => 'flag/confirm',
'title' => t('Flag confirm'),
'callback' => 'drupal_get_form',
'callback arguments' => array('flag_confirm'),
'access' => user_access('access content'),
'type' => MENU_CALLBACK,
);
}
return $items;
}
/**
* Implementation of hook_init().
*/
function flag_init() {
// Only load for non-cached pages.
if (function_exists('drupal_set_content')) {
$path = drupal_get_path('module', 'flag');
include_once $path .'/flag.inc';
if (module_exists('views')) {
include_once $path .'/includes/flag.views.inc';
}
if (module_exists('actions')) {
include_once $path .'/flag.actions.inc';
}
if (module_exists('token')) {
include_once $path .'/includes/flag.token.inc';
}
}
}
/**
* Implementation of hook_perm().
*/
function flag_perm() {
return array('administer flags');
}
/**
* Access checking to check an account for flagging ability.
*
* See documentation for $flag->user_access().
*/
function flag_access($flag, $account = NULL) {
return $flag->user_access($account);
}
/**
* Content type checking to see if a flag applies to a certain type of data.
*
* @param $flag
* The flag object whose available types are being checked.
* @param $content_type
* The type of content being checked, usually "node".
* @param $content_subtype
* The subtype (node type) being checked.
*
* @return
* Boolean TRUE if the flag is enabled for this type and subtype.
* FALSE otherwise.
*/
function flag_content_enabled($flag, $content_type, $content_subtype = NULL) {
$return = $flag->content_type == $content_type && (!isset($content_subtype) || in_array($content_subtype, $flag->types));
return $return;
}
/**
* Implementation of hook_link().
*/
function flag_link($type, $object = NULL, $teaser = FALSE) {
if (!isset($object) || !flag_fetch_definition($type)) {
return;
}
global $user;
// Anonymous users can't create flags with this system.
if (!$user->uid) {
return;
}
// Get all possible flags for this content-type.
$flags = flag_get_flags($type);
foreach ($flags as $flag) {
if (!$flag->user_access($user)) {
// User has no permission to use this flag.
continue;
}
if (!$flag->uses_hook_link($teaser)) {
// Flag is not configured to show its link here.
continue;
}
if (!$flag->applies_to_content_object($object)) {
// Flag does not apply to this content.
continue;
}
$content_id = $flag->get_content_id($object);
// The flag links are actually fully rendered theme functions.
// The HTML attribute is set to TRUE to allow whatever the themer desires.
$links['flag-'. $flag->name] = array(
'title' => $flag->theme($flag->is_flagged($content_id) ? 'unflag' : 'flag', $content_id),
'html' => TRUE,
);
}
if (isset($links)) {
return $links;
}
}
/**
* Implementation of hook_flag_link().
*
* When Flag uses a link type provided by this module, it will call this
* implementation of hook_flag_link(). It returns a single link's attributes,
* using the same structure as hook_link(). Note that "title" is provided by
* the Flag configuration if not specified here.
*
* @param $flag
* The full flag object of for the flag link being generated.
* @param $action
* The action this link will perform. Either 'flag' or 'unflag'.
* @param $content_id
* The ID of the node, comment, user, or other object being flagged.
* @return
* An array defining properties of the link.
*/
function flag_flag_link(&$flag, $action, $content_id) {
$token = flag_get_token($content_id);
return array(
'href' => "flag/". ($flag->link_type == 'confirm' ? 'confirm/' : '') ."$action/$flag->name/$content_id",
'query' => drupal_get_destination() . ($flag->link_type == 'confirm' ? '' : '&token='. $token),
);
}
/**
* Implementation of hook_flag_link_types().
*/
function flag_flag_link_types() {
return array(
'toggle' => t('JavaScript toggle'),
'normal' => t('Normal link'),
'confirm' => t('Confirmation form'),
);
}
/**
* Implementation of hook_form_alter().
*/
function flag_form_alter($form_id, &$form) {
global $user;
if ($form_id == 'node_type_form') {
$flags = flag_get_flags('node', $form['#node_type']->type, $user);
foreach ($flags as $flag) {
if ($flag->show_on_form) {
$var = 'flag_'. $flag->name .'_default';
$form['workflow']['flag'][$var] = array(
'#type' => 'checkbox',
'#title' => $flag->get_label('flag_short'),
'#default_value' => variable_get($var .'_'. $form['#node_type']->type, 0),
'#return_value' => 1,
);
}
}
if (isset($form['workflow']['flag'])) {
$form['workflow']['flag'] += array(
'#type' => 'item',
'#title' => t('Default flags'),
'#description' => t('Above are the flags you elected to show on the node editing form. You may specify their initial state here.', array('@flag-url' => url('admin/build/flags'))),
// Make the spacing a bit more compact:
'#prefix' => '
',
'#suffix' => '
',
);
}
}
elseif (isset($form['type']) && isset($form['#node'])
&& ($form_id == $form['type']['#value'] .'_node_form')) {
if (!$user->uid) {
return;
}
$nid = !empty($form['nid']['#value']) ? $form['nid']['#value'] : NULL;
$flags = flag_get_flags('node', $form['type']['#value'], $user);
// Filter out flags which need to be included on the node form.
foreach ($flags as $name => $flag) {
if (!$flag->show_on_form) {
unset($flags[$name]);
}
}
if (count($flags)) {
$form['flag'] = array(
'#type' => 'fieldset',
'#weight' => 1,
'#tree' => TRUE,
'#title' => t('Flags'),
'#collapsible' => TRUE,
);
}
foreach ($flags as $flag) {
if (isset($form['#node']->flag[$flag->name])) {
$flag_status = $form['#node']->flag[$flag->name];
}
else {
$flag_status_default = variable_get('flag_' . $flag->name . '_default_' . $form['type']['#value'], 0);
$flag_status = $nid ? $flag->is_flagged($nid) : $flag_status_default;
}
$form['flag'][$flag->name] = array(
'#type' => 'checkbox',
'#title' => $flag->get_label('flag_short', $nid),
'#description' => $flag->get_label('flag_long', $nid),
'#default_value' => $flag_status,
'#return_value' => 1,
);
}
}
}
/**
* Implementation of hook_nodeapi().
*/
function flag_nodeapi(&$node, $op, $teaser = NULL, $page = NULL) {
global $user;
switch ($op) {
case 'update':
case 'insert':
// Response to the flag checkboxes added to the form in flag_form_alter().
$remembered = FALSE;
if (isset($node->flag)) {
foreach ($node->flag as $name => $state) {
$flag = flag_get_flag($name);
// Flagging may trigger actions. We want actions to get the current
// node, not a stale database-loaded one:
if (!$remembered) {
$flag->remember_content($node->nid, $node);
// Actions may modify a node, and we don't want to overwrite this
// modification:
$remembered = TRUE;
}
flag($state ? 'flag' : 'unflag', $name, $node->nid);
}
}
break;
case 'delete':
foreach (flag_get_flags('node') as $flag) {
// If the flag is being tracked by translation set and the node is part
// of a translation set, don't delete the flagging record.
// Instead, data will be updated in the 'translation_change' op, below.
if (!$flag->i18n || empty($node->tnid)) {
db_query("DELETE FROM {flag_content} WHERE fid = %d AND content_id = %d", $flag->fid, $node->nid);
db_query("DELETE FROM {flag_counts} WHERE fid = %d AND content_id = %d", $flag->fid, $node->nid);
}
}
break;
case 'translation_change':
if (isset($node->translation_change)) {
// If there is only one node remaining, track by nid rather than tnid.
// Otherwise, use the new tnid.
$content_id = $node->translation_change['new_tnid'] == 0 ? $node->translation_change['remaining_nid'] : $node->translation_change['new_tnid'];
foreach (flag_get_flags('node') as $flag) {
if ($flag->i18n) {
db_query("UPDATE {flag_content} SET content_id = %d WHERE fid = %d AND content_id = %d", $content_id, $flag->fid, $node->translation_change['old_tnid']);
db_query("UPDATE {flag_counts} SET content_id = %d WHERE fid = %d AND content_id = %d", $content_id, $flag->fid, $node->translation_change['old_tnid']);
}
}
}
break;
}
}
/**
* Implementation of hook_user().
*/
function flag_user($op, &$edit, &$account, $category = NULL) {
switch ($op) {
case 'delete':
// Remove flags by this user.
db_query("DELETE FROM {flag_content} WHERE uid = %d", $account->uid);
break;
case 'view';
$items = array();
$flags = flag_get_flags('user');
foreach ($flags as $flag) {
if (!$flag->user_access()) {
// User has no permission to use this flag.
continue;
}
if (!$flag->applies_to_content_object($account)) {
// Flag does not apply to this content.
continue;
}
$items[$flag->get_title($account->uid)] = array(
array(
'value' => $flag->theme($flag->is_flagged($account->uid) ? 'unflag' : 'flag', $account->uid),
'class' => 'profile-' . $flag->name,
),
);
}
return $items;
}
}
/**
* Implementation of hook_node_type().
*/
function flag_node_type($op, $info) {
switch ($op) {
case 'delete':
// Remove entry from flaggable content types.
db_query("DELETE FROM {flag_types} WHERE type = '%s'", $info->type);
break;
}
}
// ---------------------------------------------------------------------------
// Administrative pages
/**
* Flag administration page. Display a list of existing flags.
*/
function flag_admin_page() {
$flags = flag_get_flags();
$default_flags = flag_get_default_flags(TRUE);
return theme('flag_admin_page', $flags, $default_flags);
}
/**
* Theme the output for the main flag administration page.
*/
function theme_flag_admin_page($flags, $default_flags) {
$output = '' . t('This page lists all the flags that are currently defined on this system. You may add new flags.', array('@add-url' => url('admin/build/flags/add'))) . '
';
// Build out the list of normal, database flags.
foreach ($flags as $flag) {
$ops = theme('links', array(
'flags_edit' => array('title' => t('edit'), 'href' => "admin/build/flags/edit/". $flag->name),
'flags_delete' => array('title' => t('delete'), 'href' => "admin/build/flags/delete/". $flag->name),
));
$roles = array_flip(array_intersect(array_flip(user_roles()), $flag->roles));
$rows[] = array(
$flag->name,
$flag->content_type,
empty($flag->roles) ? '' . t('No roles') . '' : implode(', ', $roles),
$flag->types ? implode(', ', $flag->types) : '-',
$flag->global ? t('Yes') : t('No'),
$ops,
);
}
if (!$flags) {
$rows[] = array(
array('data' => t('No flags are currently defined.'), 'colspan' => 6),
);
}
$header = array(t('Flag'), t('Flag type'), t('Roles'), t('Node types'), t('Global?'), t('Operations'));
$output .= theme('table', $header, $rows);
// Build a list of disabled, module-based flags.
$rows = array();
foreach ($default_flags as $name => $flag) {
if (!isset($flags[$name])) {
$ops = theme('links', array(
'flags_enable' => array('title' => t('enable'), 'href' => "admin/build/flags/edit/". $flag->name),
));
$roles = array_flip(array_intersect(array_flip(user_roles()), $flag->roles));
$rows[] = array(
$flag->name,
$flag->module,
$flag->content_type,
$ops,
);
}
}
if (!empty($rows)) {
$header = array(t('Disabled Flags'), t('Module'), t('Flag type'), t('Operations'));
$output .= theme('table', $header, $rows);
}
if (!module_exists('views')) {
$output .= '' . t('The Views module is not installed, or not enabled. It is recommended that you install the Views module to be able to easily produce lists of flagged content.', array('@views-url' => url('http://drupal.org/project/views'))) . '
';
}
else {
$output .= '';
$output .= t('Lists of flagged content can be displayed using views. You can configure these in the Views administration section.', array('@views-url' => url('admin/build/views')));
if (flag_get_flag('bookmarks')) {
$output .= ' ' . t('Flag module automatically provides a few default views for the bookmarks flag. You can use these as templates by cloning these views and then customizing as desired.', array('@views-url' => url('admin/build/views')));
}
$output .= ' ' . t('The Flag module handbook contains extensive documentation on creating customized views using flags.', array('@flag-handbook-url' => 'http://drupal.org/handbook/modules/flag', '@customize-url' => 'http://drupal.org/node/296954'));
$output .= '
';
}
if (!module_exists('flag_actions')) {
$output .= '' . t('Flagging an item may trigger actions. However, you don\'t have the Flag actions module enabled, so you won\'t be able to enjoy this feature.', array('@actions-url' => url('admin/build/flags/actions'), '@modules-url' => url('admin/build/modules'))) . '
';
}
else {
$output .= '' . t('Flagging an item may trigger actions.', array('@actions-url' => url('admin/build/flags/actions'))) . '
';
}
$output .= '' . t('To learn about the various ways to use flags, please check out the Flag module handbook.', array('@handbook-url' => 'http://drupal.org/handbook/modules/flag')) . '
';
return $output;
}
/**
* Menu callback for adding a new flag.
*/
function flag_add_page($type = NULL, $name = NULL) {
if (isset($type) && isset($name)) {
return drupal_get_form('flag_form', $name, $type);
}
else {
return drupal_get_form('flag_add_form');
}
}
/**
* Present a form for creating a new flag, setting the type of flag.
*/
function flag_add_form() {
$form = array();
$form['name'] = array(
'#type' => 'textfield',
'#title' => t('Flag name'),
'#description' => t('The machine-name for this flag. It may be up to 32 characters long and my only contain lowercase letters, underscores, and numbers. It will be used in URLs and in all API calls.'),
'#maxlength' => 32,
'#required' => TRUE,
);
$types = array();
foreach (flag_fetch_definition() as $type => $info) {
$types[$type] = $info['title'] . '' . $info['description'] . '
';
}
$form['type'] = array(
'#type' => 'radios',
'#title' => t('Flag type'),
'#default_value' => 'node',
'#description' => t('The type of content this flag will affect. An individual flag can only affect one type of content. This cannot be changed once the flag is created.'),
'#required' => TRUE,
'#options' => $types,
);
$form['drupal-5-warning'] = array(
'#type' => 'fieldset',
'#collapsible' => TRUE,
'#collapsed' => TRUE,
'#title' => t('Important note concerning the limitation of Views 1.x'),
'#description' => t('You\'re using Drupal 5.
This module utilizes Views to produce lists of flagged content. Since you\'re using Drupal 5, you will be using Views 1.x.
Since Views 1.x can only produce lists of nodes, you won\'t be able to produce lists of flagged users (or of flagged comments) with it. Therefore, while this module lets you flag users and comments, you may not find this feature useful. We have a handbook page explaining the limitations of the Drupal 5 version of this module.
Note, however, that you can still flag and list users; but indirectly:
Many Drupal 5 websites use modules such as Bio or Node Profile, that create corresponding nodes for users. So, instead of flagging users directly, you can flag these nodes. Views can easily produce a list of them.
In other words, if you\'re using Bio or Node Profile, and you want to flag users, make your flag of type "Nodes".
'),
);
$form['submit'] = array(
'#type' => 'submit',
'#value' => t('Submit'),
);
return $form;
}
function flag_add_form_validate($form_id, $form_values, $form) {
$flag = flag_flag::factory_by_content_type($form_values['type']);
$flag->name = $form_values['name'];
$flag->validate_name();
}
function flag_add_form_submit($form_id, $form_values) {
drupal_goto('admin/build/flags/add/'. $form_values['type'] .'/'. $form_values['name']);
}
/**
* Add/Edit flag page.
*/
function flag_form($name = NULL, $type = NULL) {
if (isset($type)) {
// Adding a new flag.
$flag = flag_flag::factory_by_content_type($type);
$flag->name = $name;
drupal_set_title(t('Add new flag'));
}
else {
// Editing an existing flag.
$flag = flag_get_flag($name);
if (empty($flag)) {
// Check if we're overriding a default flag.
$default_flags = flag_get_default_flags(TRUE);
if (isset($default_flags[$name])) {
$flag = $default_flags[$name];
}
else {
drupal_goto('admin/build/flags');
}
}
drupal_set_title(t('Edit @title flag', array('@title' => $flag->get_title())));
}
$form['_flag'] = array(
'#type' => 'value',
'#value' => $flag,
);
$form['name'] = array(
'#type' => 'textfield',
'#title' => t('Name'),
'#default_value' => $flag->name,
'#description' => t('The machine-name for this flag. It may be up to 32 characters long and my only contain lowercase letters, underscores, and numbers. It will be used in URLs and in all API calls.'),
'#maxlength' => 32,
'#required' => TRUE,
'#access' => empty($flag->locked['name']),
);
if (!empty($flag->fid)) {
$form['name']['#description'] .= ' '. t('Change this value only with great care.') .'';
}
$form['title'] = array(
'#type' => 'textfield',
'#title' => t('Title'),
'#default_value' => $flag->title,
'#description' => t('A short, descriptive title for this flag. It will be used in administrative interfaces to refer to this flag, and in page titles and menu items of some views this module provides (theses are customizable, though). Some examples could be Bookmarks, Favorites, or Offensive.', array('@insite-views-url' => url('admin/build/views'))),
'#maxlength' => 255,
'#required' => TRUE,
'#access' => empty($flag->locked['title']),
);
$form['flag_short'] = array(
'#type' => 'textfield',
'#title' => t('Flag link text'),
'#default_value' => $flag->flag_short,
'#description' => t('The text for the "flag this" link for this flag.'),
'#required' => TRUE,
'#access' => empty($flag->locked['flag_short']),
);
$form['flag_long'] = array(
'#type' => 'textfield',
'#title' => t('Flag link description'),
'#default_value' => $flag->flag_long,
'#description' => t('The description of the "flag this" link. Usually displayed on mouseover.'),
'#access' => empty($flag->locked['flag_long']),
);
$form['flag_confirmation'] = array(
'#type' => 'textfield',
'#title' => t('Flag confirmation message'),
'#default_value' => $flag->flag_confirmation,
'#description' => t('Message displayed if the user has clicked the "flag this" link and confirmation is required. Usually presented in the form of a question such as, "Are you sure you want to flag this content?"'),
'#access' => empty($flag->locked['flag_confirmation']),
);
$form['flag_message'] = array(
'#type' => 'textfield',
'#title' => t('Flagged message'),
'#default_value' => $flag->flag_message,
'#description' => t('Message displayed after flagging content. If JavaScript is enabled, it will be displayed below the link. If not, it will be displayed in the message area.'),
'#access' => empty($flag->locked['flag_message']),
);
$form['unflag_short'] = array(
'#type' => 'textfield',
'#title' => t('Unflag link text'),
'#default_value' => $flag->unflag_short,
'#description' => t('The text for the "unflag this" link for this flag.'),
'#required' => TRUE,
'#access' => empty($flag->locked['unflag_short']),
);
$form['unflag_long'] = array(
'#type' => 'textfield',
'#title' => t('Unflag link description'),
'#default_value' => $flag->unflag_long,
'#description' => t('The description of the "unflag this" link. Usually displayed on mouseover.'),
'#access' => empty($flag->locked['unflag_long']),
);
$form['unflag_confirmation'] = array(
'#type' => 'textfield',
'#title' => t('Unflag confirmation message'),
'#default_value' => $flag->unflag_confirmation,
'#description' => t('Message displayed if the user has clicked the "unflag this" link and confirmation is required. Usually presented in the form of a question such as, "Are you sure you want to unflag this content?"'),
'#access' => empty($flag->locked['unflag_confirmation']),
);
$form['unflag_message'] = array(
'#type' => 'textfield',
'#title' => t('Unflagged message'),
'#default_value' => $flag->unflag_message,
'#description' => t('Message displayed after content has been unflagged. If JavaScript is enabled, it will be displayed below the link. If not, it will be displayed in the message area.'),
'#access' => empty($flag->locked['unflag_message']),
);
if (module_exists('token')) {
$form['token_help'] = array(
'#title' => t('Token replacement'),
'#type' => 'fieldset',
'#description' => t('The above six options may contain the following wildcard replacements. For example, "Mark Link" could be entered as "Add [title] to your flags" or "Add this [type-name] to your flags". These wildcards will be replaced with the appropriate field from the node.') . theme('flag_token_help', $flag->get_labels_token_types()),
'#collapsible' => TRUE,
'#collapsed' => TRUE,
);
}
else {
$form['token_help'] = array(
'#value' => '' . t('Note: You don\'t have the Token module installed. If you have it installed, and enabled, you\'ll be able to embed tokens in the six labels above.', array('@token-url' => 'http://drupal.org/project/token')) . '',
);
}
$form['global'] = array(
'#type' => 'checkbox',
'#title' => t('Global flag'),
'#default_value' => $flag->global,
'#description' => t('If checked, flag is considered "global" and each node is either flagged or not. If unchecked, each user has individual flags on content.'),
'#weight' => 1,
'#access' => empty($flag->locked['global']),
);
$form['roles'] = array(
'#type' => 'checkboxes',
'#title' => t('Roles that may use this flag'),
'#options' => user_roles(TRUE),
'#default_value' => $flag->roles,
'#required' => TRUE,
'#description' => t('Checking authenticated user will allow all logged-in users to flag content with this flag. Anonymous users may not flag content.'),
'#weight' => 5,
'#access' => empty($flag->locked['roles']),
);
// Disabled access breaks checkboxes unless #value is hard coded.
if (!empty($flag->locked['roles'])) {
$form['roles']['#value'] = $flag->roles;
}
$form['types'] = array(
'#type' => 'checkboxes',
'#title' => t('What nodes this flag may be used on'),
'#options' => node_get_types('names'),
'#default_value' => $flag->types,
'#description' => t('Check any node types that this flag may be used on. You must check at least one node type.'),
'#required' => TRUE,
'#weight' => 10,
'#access' => empty($flag->locked['types']),
);
// Disabled access breaks checkboxes unless #value is hard coded.
if (!empty($flag->locked['types'])) {
$form['types']['#value'] = $flag->types;
}
$form['display'] = array(
'#type' => 'fieldset',
'#title' => t('Display options'),
'#description' => t('Flags are usually controlled through links that allow users to toggle their behavior. You can choose how users interact with flags by changing options here. It is legitimate to have none of the following checkboxes ticked, if, for some reason, you wish to place the the links on the page yourself.', array('@placement-url' => 'http://drupal.org/node/295383')),
'#tree' => FALSE,
'#weight' => 20,
);
$form['display']['link_type'] = array(
'#type' => 'radios',
'#title' => t('Link type'),
'#options' => _flag_link_type_options(),
'#default_value' => $flag->link_type,
'#weight' => 2,
'#access' => empty($flag->locked['link_type']),
);
$form['submit'] = array(
'#type' => 'submit',
'#value' => t('Submit'),
// We put this button on the form before calling $flag->options_form()
// to give the flag handler a chance to remove it (e.g. flag_broken).
'#weight' => 999,
);
$flag->options_form($form);
return $form;
}
/**
* Validate required checkboxes on our form.
*
* FAPI has a problem validating #required checkboxes: if previously non-empty
* checkboxes are cleared, FAPI won't detect this.
*/
function flag_form_validate($form_id, $form_values, $form) {
// Fix checkboxes.
foreach (element_children($form) as $field) {
if ($form[$field]['#type'] == 'checkboxes' && $form[$field]['#required'] && !array_filter($form_values[$field])) {
form_error($form[$field], t('!name field is required.', array('!name' => $form[$field]['#title'])));
}
}
if ($form_values['link_type'] == 'confirm') {
if (empty($form_values['flag_confirmation'])) {
form_set_error('flag_confirmation', t('A flag confirmation message is required when using the confirmation link type.'));
}
if (empty($form_values['unflag_confirmation'])) {
form_set_error('unflag_confirmation', t('An unflag confirmation message is required when using the confirmation link type.'));
}
}
$flag = $form_values['_flag'];
unset($form_values['_flag']);
$flag->form_input($form_values);
$flag->validate();
}
/**
* Add/Edit flag form submit.
*/
function flag_form_submit($form_id, $form_values) {
$flag = $form_values['_flag'];
unset($form_values['_flag']);
$flag->form_input($form_values);
$flag->save();
$flag->enable();
drupal_set_message(t('Flag @name has been saved.', array('@name' => $flag->get_title())));
_flag_clear_cache();
return 'admin/build/flags';
}
/**
* Delete flag page.
*/
function flag_delete_confirm($name) {
$flag = flag_get_flag($name);
if (empty($flag)) {
drupal_goto('admin/build/flags');
}
$form['fid'] = array('#type' => 'value', '#value' => $flag->fid);
return confirm_form($form,
t('Are you sure you want to delete %title?', array('%title' => $flag->get_title())),
!empty($_GET['destination']) ? $_GET['destination'] : 'admin/build/flags',
isset($flag->module) ? t('This flag is provided by the @module module. It will loose any customizations and be disabled.', array('@module' => node_get_types('name', $flag->module))) : t('This action cannot be undone.'),
t('Delete'), t('Cancel')
);
}
function flag_delete_confirm_submit($form_id, $form_values) {
$flag = flag_get_flag(NULL, $form_values['fid']);
if ($form_values['confirm']) {
$flag->delete();
$flag->disable();
_flag_clear_cache();
}
drupal_set_message(t('Flag @name has been deleted.', array('@name' => $flag->get_title())));
return 'admin/build/flags';
}
/*
* Clears various caches when a flag is modified.
*/
function _flag_clear_cache() {
if (module_exists('views')) {
views_invalidate_cache();
}
// Reset our flags cache, thereby making the following code aware of the
// modifications.
flag_get_flags(NULL, NULL, NULL, TRUE);
// The title of a flag may appear in the menu (indirectly, via our "default
// views"), so we need to clear the menu cache. This call also clears the
// page cache, which is desirable too because the flag labels may have
// changed.
menu_rebuild();
}
/**
* Menu callback for (un)flagging a node.
*
* Used both for the regular callback as well as the JS version.
*/
function flag_page($action, $flag_name, $content_id) {
$js = isset($_REQUEST['js']);
$token = $_REQUEST['token'];
// Check the flag token, then perform the flagging.
if (!flag_check_token($token, $content_id)) {
$error = t('Bad token. You seem to have followed an invalid link.');
}
else {
$result = flag($action, $flag_name, $content_id);
if (!$result) {
$error = t('You are not allowed to flag, or unflag, this content.');
}
}
// If an error was received, set a message and exit.
if (isset($error)) {
if ($js) {
drupal_set_header('Content-Type: text/javascript; charset=utf-8');
print drupal_to_js(array(
'status' => FALSE,
'errorMessage' => $error,
));
exit;
}
else {
drupal_set_message($error);
drupal_access_denied();
return;
}
}
// If successful, return data according to the request type.
if ($js) {
drupal_set_header('Content-Type: text/javascript; charset=utf-8');
$flag = flag_get_flag($flag_name);
$flag->link_type = 'toggle';
print drupal_to_js(array(
'status' => TRUE,
'newLink' => $flag->theme($flag->is_flagged($content_id) ? 'unflag' : 'flag', $content_id, TRUE),
// Further information for the benefit of custom JavaScript event handlers:
'contentId' => $content_id,
'contentType' => $flag->content_type,
'flagName' => $flag->name,
'flagStatus' => $flag->is_flagged($content_id) ? 'flagged' : 'unflagged',
));
exit;
}
else {
$flag = flag_get_flag($flag_name);
drupal_set_message($flag->get_label($action . '_message', $content_id));
drupal_goto();
}
}
/**
* Form for confirming the (un)flagging of a piece of content.
*/
function flag_confirm($action = 'flag', $flag_name, $content_id = '') {
$form = array();
$form['action'] = array(
'#type' => 'value',
'#value' => $action,
);
$form['flag_name'] = array(
'#type' => 'value',
'#value' => $flag_name,
);
$form['content_id'] = array(
'#type' => 'value',
'#value' => $content_id,
);
$flag = flag_get_flag($flag_name);
$question = $flag->get_label($action .'_confirmation', $content_id);
$path = isset($_GET['destination']) ? $_GET['destination'] : '';
$yes = $flag->get_label($action .'_short', $content_id);
return confirm_form($form, $question, $path, '', $yes);
}
function flag_confirm_submit($form_id, $form_values) {
$action = $form_values['action'];
$flag_name = $form_values['flag_name'];
$content_id = $form_values['content_id'];
$result = flag($action, $flag_name, $content_id);
if (!$result) {
drupal_set_message(t('You are not allowed to flag, or unflag, this content.'));
}
else {
$flag = flag_get_flag($flag_name);
drupal_set_message($flag->get_label($action . '_message', $content_id));
}
}
/**
* Flags or unflags an item.
*
* @param $account
* The user on whose behalf to flag. Leave empty for the current user.
* @return
* FALSE if some error occured (e.g., user has no permission, flag isn't
* applicable to the item, etc.), TRUE otherwise.
*/
function flag($action, $flag_name, $content_id, $account = NULL) {
if (!($flag = flag_get_flag($flag_name))) {
// Flag does not exist.
return FALSE;
}
return $flag->flag($action, $content_id, $account);
}
/**
* Implementation of hook_flag(). Trigger actions if any are available.
*/
function flag_flag($action, $flag, $content_id, $account) {
if (module_exists('actions') && function_exists('_actions_get_hook_aids')) {
$flag_action = $flag->get_flag_action($content_id);
$flag_action->action = $action;
$context = (array)$flag_action;
$aids = _actions_get_hook_aids($action, $action);
foreach ($aids as $aid => $action_info) {
// The 'if ($aid)' is a safeguard against http://drupal.org/node/271460#comment-886564
if ($aid) {
actions_do($aid, $flag, $context);
}
}
}
}
/**
* Implementation of hook_node_operations().
*
* Add additional options on the admin/build/node page.
*/
function flag_node_operations() {
global $user;
$flags = flag_get_flags('node', NULL, $user);
$operations = array();
foreach ($flags as $flag) {
$operations['flag_'. $flag->name] = array(
'label' => $flag->get_label('flag_short'),
'callback' => 'flag_nodes',
'callback arguments' => array('flag', $flag->name),
);
$operations['unflag_'. $flag->name] = array(
'label' => $flag->get_label('unflag_short'),
'callback' => 'flag_nodes',
'callback arguments' => array('unflag', $flag->name),
);
}
return $operations;
}
/**
* Callback function for hook_node_operations().
*/
function flag_nodes($nodes, $action, $flag_name) {
foreach ($nodes as $nid) {
flag($action, $flag_name, $nid);
}
}
/**
* Theme an individual link and a message after the link is marked.
*
* @param $flag
* The flag object.
* @param $action
* Which link to show: either "flag" or "unflag".
* @param $content_id
* The ID of the content being flagged.
* @param $after_flagging
* This function is called for both the link both before and after being
* flagged. If displaying to the user immediately after flagging, this value
* will be boolean TRUE. This is usually used in conjunction with immedate
* JavaScript-based toggling of flags.
*/
function theme_flag($flag, $action, $content_id, $after_flagging = FALSE) {
static $template;
if (!isset($template)) {
$template = './' . drupal_get_path('module', 'flag') . '/theme/flag.tpl.php';
}
$variables = array(
'flag' => $flag,
'action' => $action,
'content_id' => $content_id,
'after_flagging' => $after_flagging,
);
flag_template_preprocess('flag', $variables);
extract($variables);
ob_start();
include $template;
$output = ob_get_contents();
ob_end_clean();
// Unfortunately, the antique jQuery shipped with Drupal 5 has a bug: the
// expression $('something') fails is 'something' contains a
// newline. That's because jQuery's source has:
//
// // Handle HTML strings
// if ( typeof a == "string" ) {
// var m = /^[^<]*(<.+>)[^>]*$/.exec(a);
// if ( m ) a = jQuery.clean( [ m[1] ] );
// }
//
// Note the "<.+>", which doesn't match newlines. In newer jQuery's that
// expression was changed to:
//
// var quickExpr = /^[^<]*(<(.|\s)+>)[^>]*$|^#(\w+)$/
//
// Note the \s which does match newlines.
//
// So we have to remove all newlines:
$output = strtr($output, "\r\n", ' ');
return $output;
}
/**
* A preprocess function for our theme('flag'). It generates the
* variables needed there.
*
* See 'flag.tpl.php' for documentation for these variables.
*
* Note: The Drupal 5 version of this module calls this function directly.
*/
function template_preprocess_flag(&$variables) {
static $first_time = TRUE;
// Some typing shotcuts:
$flag =& $variables['flag'];
$action = $variables['action'];
$content_id = $variables['content_id'];
// Generate the link URL.
$link_types = flag_get_link_types();
$link_type_module = $link_types[$flag->link_type]['module'];
$link = module_invoke($link_type_module, 'flag_link', $flag, $action, $content_id);
if (isset($link['title']) && empty($link['html'])) {
$link['title'] = check_plain($link['title']);
}
if ($flag->link_type == 'toggle' && $first_time) {
$variables['setup'] = $first_time;
$first_time = FALSE;
}
else {
$variables['setup'] = FALSE;
}
$variables['link_href'] = check_url(url($link['href'], $link['query'], $link['fragment']));
$variables['link_text'] = isset($link['title']) ? $link['title'] : strip_tags($flag->get_label($action . '_short', $content_id), '');
$variables['link_title'] = isset($link['attributes']['title']) ? $link['attributes']['title'] : strip_tags($flag->get_label($action . '_long', $content_id));
$variables['flag_name_css'] = str_replace('_', '-', $flag->name);
$variables['last_action'] = ($action == 'flag' ? 'unflagged' : 'flagged');
$variables['flag_classes_array'] = array();
$variables['flag_classes_array'][] = 'flag';
$variables['flag_classes_array'][] = $variables['action'] .'-action';
$variables['flag_classes_array'][] = 'flag-link-'. $flag->link_type;
if (isset($link['attributes']['class'])) {
$variables['flag_classes_array'][] = $link['attributes']['class'];
}
if ($variables['after_flagging']) {
$inverse_action = ($action == 'flag' ? 'unflag' : 'flag');
$variables['message_text'] = $flag->get_label($inverse_action . '_message', $content_id);
$variables['flag_classes_array'][] = $variables['last_action'];
}
$variables['flag_classes'] = implode(' ', $variables['flag_classes_array']);
}
/**
* In Drupal 5 it is possible to put a phptemplate_flag() function in your
* 'template.php', and within that function instructions to make Drupal scan
* for various template files for theming a flag.
*
* The syntax for that phptemplate_flag() is somewhat arcane --and may need
* updating when our API changes-- so here we provide a ready-made version of
* that funtion. You should call it from your phptemplate_flag(). See
* 'theme/README.txt'.
*/
function flag_phptemplate_adapter($flag, $action, $content_id, $after_flagging = FALSE) {
// Prepare the variables to be available in the template.
$variables = array(
'flag' => $flag,
'action' => $action,
'content_id' => $content_id,
'after_flagging' => $after_flagging,
);
flag_template_preprocess('flag', $variables);
// Prepare an array of suggested templates to try.
$suggestions = array();
foreach ($flag->theme_suggestions() as $suggestion) {
$suggestions[] = str_replace('_', '-', $suggestion);
}
$suggestions = array_reverse($suggestions);
// Hand control to the phptemplate engine.
$output = _phptemplate_callback('flag', $variables, $suggestions);
// Finally, handle Drupal 5's jQuery bug. See explanation in theme_flag().
$output = strtr($output, "\r\n", ' ');
return $output;
}
/**
* Call module flag preprocess functions for Drupal 5.
*
* The preprocess functions for themes and phptemplate can be done in
* _phptemplate_variables(), but this adds preprocessing to modules that don't
* have this access in Drupal 5.
*
* @param $hook
* The name of the template file hook whose variables are being preprocessed.
* @param $variables
* The existing variables that will be passed to the template file.
*/
function flag_template_preprocess($hook, &$variables) {
if (function_exists('template_preprocess_'. $hook)) {
$function = 'template_preprocess_'. $hook;
$function($variables);
}
foreach (module_implements('preprocess_'. $hook) as $module) {
$function = $module .'_preprocess_'. $hook;
$function($variables);
}
}
/**
* Format a string containing a count of items.
*
* _flag_format_plural() is a version of format_plural() which
* accepts the format string as a single argument, where the singular and
* plural forms are separated by pipe. A 'zero' form is allowed as well.
*
* _flag_format_plural() is used where we want the admin, not the
* programmer, to be able to nicely and easily format a number.
*
* If three forms are provided, separated by pipes, then the first is
* considered the zero form and is used if $count is 0. The zero form may
* well be an empty string.
*
* @param $count
* The item count to display.
* @param $format
* The singular, plural, and optionally the zero, forms separated
* by pipe characters.
* @return
* A formatted, translated string.
*
* Examples for $format:
*
* @code
* "@count"
* "1 vote|@count votes"
* "needs voting|1 vote|@count votes"
* "|1 vote|@count votes"
* "|@count|@count"
* @endcode
*/
function _flag_format_plural($count, $format) {
$elements = explode('|', $format ? $format : '@count', 3);
if (count($elements) == 3) {
list($zero, $singular, $plural) = $elements;
}
elseif (count($elements) == 2) {
list($singular, $plural) = $elements;
$zero = NULL;
}
else { // count($elements) == 1
$singular = $plural = $elements[0];
$zero = NULL;
}
if (isset($zero) && intval($count) == 0) {
return $zero;
}
else {
return format_plural(intval($count), $singular, $plural);
}
}
/**
* Return an array of flag names keyed by fid.
*/
function _flag_get_flag_names() {
$flags = flag_get_flags();
$flag_names = array();
foreach ($flags as $flag) {
$flag_names[$flag->fid] = $flag->name;
}
return $flag_names;
}
/**
* Return an array of flag link types suitable for a select list or radios.
*/
function _flag_link_type_options() {
$options = array();
$types = flag_get_link_types();
foreach ($types as $type_name => $type) {
$options[$type_name] = $type['title'];
}
return $options;
}
// ---------------------------------------------------------------------------
// Non-Views public API
/**
* Get flag counts for all flags on a node.
*
* @param $content_type
* The content type (usually 'node').
* @param $content_id
* The content ID (usually the node ID).
* @param $reset
* Reset the internal cache and execute the SQL query another time.
*
* @return $flags
* An array of the structure [name] => [number of flags].
*/
function flag_get_counts($content_type, $content_id, $reset = FALSE) {
static $counts;
if ($reset) {
$counts = array();
if (!isset($content_type)) {
return;
}
}
if (!isset($counts[$content_type][$content_id])) {
$counts[$content_type][$content_id] = array();
$result = db_query("SELECT f.name, fc.count FROM {flags} f LEFT JOIN {flag_counts} fc ON f.fid = fc.fid WHERE fc.content_type = '%s' AND fc.content_id = %d", $content_type, $content_id);
while ($row = db_fetch_object($result)) {
$counts[$content_type][$content_id][$row->name] = $row->count;
}
}
return $counts[$content_type][$content_id];
}
/**
* Load a single flag either by name or by flag ID.
*
* @param $name
* The the flag name.
* @param $fid
* The the flag id.
*/
function flag_get_flag($name = NULL, $fid = NULL) {
$flags = flag_get_flags();
if (isset($name)) {
if (isset($flags[$name])) {
return $flags[$name];
}
}
elseif (isset($fid)) {
foreach ($flags as $flag) {
if ($flag->fid == $fid) {
return $flag;
}
}
}
}
/**
* List all flags available.
*
* If node type or account are entered, a list of all possible flags will be
* returned.
*
* @param $content_type
* Optional. The type of content for which to load the flags. Usually 'node'.
* @param $content_subtype
* Optional. The node type for which to load the flags.
* @param $account
* Optional. The user accont to filter available flags. If not set, all
* flags for will this node will be returned.
* @param $reset
* Optional. Reset the internal query cache.
*
* @return $flags
* An array of the structure [fid] = flag_object.
*/
function flag_get_flags($content_type = NULL, $content_subtype = NULL, $account = NULL, $reset = FALSE) {
static $flags;
// Retrieve a list of all flags, regardless of the parameters.
if (!isset($flags) || $reset) {
$flags = array();
// Database flags.
$result = db_query("SELECT f.*, fn.type FROM {flags} f LEFT JOIN {flag_types} fn ON fn.fid = f.fid");
while ($row = db_fetch_object($result)) {
if (!isset($flags[$row->name])) {
$flags[$row->name] = flag_flag::factory_by_row($row);
}
else {
$flags[$row->name]->types[] = $row->type;
}
}
// Add code-based flags provided by modules.
$default_flags = flag_get_default_flags();
foreach ($default_flags as $name => $default_flag) {
// Insert new enabled flags into the database to give them an FID.
if ($default_flag->status && !isset($flags[$name])) {
$default_flag->save();
$flags[$name] = $default_flag;
}
if (isset($flags[$name])) {
// Ensure overridden flags are associated with their parent module.
$flags[$name]->module = $default_flag->module;
// Update the flag with any properties that are "locked" by the code version.
if (isset($default_flag->locked)) {
$flags[$name]->locked = $default_flag->locked;
foreach ($default_flag->locked as $property) {
$flags[$name]->$property = $default_flag->$property;
}
}
}
}
}
// Make a variable copy to filter types and account.
$filtered_flags = $flags;
// Filter out flags based on type and subtype.
if (isset($content_type) || isset($content_subtype)) {
foreach ($filtered_flags as $name => $flag) {
if (!flag_content_enabled($flag, $content_type, $content_subtype)) {
unset($filtered_flags[$name]);
}
}
}
// Filter out flags based on account permissions.
if (isset($account) && $account->uid != 1) {
foreach ($filtered_flags as $name => $flag) {
if (!flag_access($flag, $account)) {
unset($filtered_flags[$name]);
}
}
}
return $filtered_flags;
}
/**
* Retrieve a list of flags defined by modules.
*
* @param $include_disabled
* Unless specified, only enabled flags will be returned.
* @return
* An array of flag prototypes, not usable for flagging. Use flag_get_flags()
* if needing to perform a flagging with any enabled flag.
*/
function flag_get_default_flags($include_disabled = FALSE) {
$default_flags = array();
$flag_status = variable_get('flag_default_flag_status', array());
foreach (module_implements('flag_default_flags') as $module) {
$function = $module . '_flag_default_flags';
foreach ($function() as $config) {
$flag = flag_flag::factory_by_array($config);
$flag->module = $module;
// Add flags that have been enabled.
if ((!isset($flag_status[$flag->name]) && (!isset($flag->status) || $flag->status)) || !empty($flag_status[$flag->name])) {
$flag->status = TRUE;
$default_flags[$flag->name] = $flag;
}
elseif ($include_disabled) {
$flag->status = FALSE;
$default_flags[$flag->name] = $flag;
}
}
}
return $default_flags;
}
/**
* Find what a user has flagged, either a single node or on the entire site.
*
* @param $content_type
* The type of content that will be retrieved. Usually 'node'.
* @param $content_id
* Optional. The content ID to check for flagging. If none given, all
* content flagged by this user will be returned.
* @param $uid
* Optional. The user ID whose flags we're checking. If none given, the
* current user will be used.
* @param $reset
* Reset the internal cache and execute the SQL query another time.
*
* @return $flags
* If returning a single node's flags, an array of the structure
* [name] => (fid => [fid], uid => [uid] nid => [nid], timestamp => [timestamp])
*
* If returning all nodes, an array of the arrays for each node.
* [nid] => [name] => Object from above.
*
*/
function flag_get_user_flags($content_type, $content_id = NULL, $uid = NULL, $reset = FALSE) {
static $flagged_content;
if ($reset) {
$flagged_content = array();
if (!isset($content_type)) {
return;
}
}
$uid = !isset($uid) ? $GLOBALS['user']->uid : $uid;
if (isset($content_id)) {
if (!isset($flagged_content[$uid][$content_type][$content_id])) {
$flag_names = _flag_get_flag_names();
$flagged_content[$uid][$content_type][$content_id] = array();
$result = db_query("SELECT * FROM {flag_content} WHERE content_type = '%s' AND content_id = %d AND (uid = %d OR uid = 0)", $content_type, $content_id, $uid);
while ($flag_content = db_fetch_object($result)) {
$flagged_content[$uid][$content_type][$content_id][$flag_names[$flag_content->fid]] = $flag_content;
}
}
return $flagged_content[$uid][$content_type][$content_id];
}
else {
if (!isset($flagged_content[$uid]['all'][$content_type])) {
$flag_names = _flag_get_flag_names();
$flagged_content[$uid]['all'][$content_type] = TRUE;
$result = db_query("SELECT * FROM {flag_content} WHERE content_type = '%s' AND (uid = %d OR uid = 0)", $content_type, $uid);
while ($flag_content = db_fetch_object($result)) {
$flagged_content[$uid][$content_type]['all'][$flag_names[$flag_content->fid]][$flag_content->content_id] = $flag_content;
}
}
return $flagged_content[$uid][$content_type]['all'];
}
}
/**
* Return a list of users who have flagged a piece of content.
*/
function flag_get_content_flags($content_type, $content_id, $reset = FALSE) {
static $content_flags;
if (!isset($content_flags[$content_type][$content_id]) || $reset) {
$flag_names = _flag_get_flag_names();
$result = db_query("SELECT * FROM {flag_content} WHERE content_type = '%s' AND content_id = %d ORDER BY timestamp DESC", $content_type, $content_id);
while ($flag_content = db_fetch_object($result)) {
$content_flags[$content_type][$content_id]['users'][$flag_content->uid][$flag_names[$flag_content->fid]] = $flag_content;
}
}
return $content_flags[$content_type][$content_id]['users'];
}
/**
* A utility function for outputting a flag link.
*
* You should call this function from your template when you want to put the
* link on the page yourself. For example, you could call this function from
* your 'node.tpl.php':
*
* nid); ?>
*
* @param $flag_name
* The "machine readable" name of the flag; e.g. 'bookmarks'.
* @param $content_id
* The content ID to check for flagging. This is usually a node ID.
*/
function flag_create_link($flag_name, $content_id) {
$flag = flag_get_flag($flag_name);
if (!$flag) {
// Flag does not exist.
return;
}
if (!$flag->user_access()) {
// User has no permission to use this flag.
return;
}
if (!$flag->applies_to_content_id($content_id)) {
// Flag does not apply to this content.
return;
}
return $flag->theme($flag->is_flagged($content_id) ? 'unflag' : 'flag', $content_id);
}
/**
* Return an array of link types provided by modules.
*/
function flag_get_link_types($reset = FALSE) {
static $link_types;
if (!isset($link_types) || $reset) {
$link_types = array();
foreach (module_implements('flag_link_types') as $module) {
$module_types = module_invoke($module, 'flag_link_types');
foreach ($module_types as $type_name => $type_title) {
$link_types[$type_name] = array(
'module' => $module,
'title' => $type_title,
);
}
}
}
return $link_types;
}
/**
* Get a private token used to protect links from spoofing - CSRF.
*/
function flag_get_token($nid) {
return drupal_get_token($nid);
}
/**
* Check to see if a token value matches the specified node.
*/
function flag_check_token($token, $seed) {
return drupal_get_token($seed) == $token;
}