Skip to content
views_rss_core.module 13.8 KiB
Newer Older
 * @file
 * Provides core <channel> and <item> elements for Views RSS module.
/**
 * Module installation path.
 */
define('VIEWS_RSS_CORE_PATH', drupal_get_path('module', 'views_rss_core'));

/**
 * Include file with field formatters.
 */
include_once VIEWS_RSS_CORE_PATH .'/views_rss_core.field.inc';

/**
 * Include file with all preprocess functions.
 */
include_once VIEWS_RSS_CORE_PATH . '/views_rss_core.inc';
/**
 * Implementation of hook_views_rss_namespaces().
 */
function views_rss_core_views_rss_namespaces() {
  $namespaces['atom'] = array(
    'prefix' => 'xmlns',
    'uri' => 'http://www.w3.org/2005/Atom',
  );
/**
 * Implementation of hook_views_rss_channel_elements().
 */
function views_rss_core_views_rss_channel_elements() {
  $elements['title'] = array(
    'configurable' => FALSE,
    'preprocess functions' => array(
      'views_rss_core_preprocess_channel_title',
      'views_rss_htmlspecialchars',
    ),
    'help' => 'http://www.rssboard.org/rss-profile#element-channel-title',
  );
  $elements['description'] = array(
    'description' => t('Description for this feed. If left blank, the default site mission will be used.'),
    'settings form' => array('#type' => 'textarea', '#rows' => 3),
    'preprocess functions' => array(
      'views_rss_core_preprocess_channel_description',
      'views_rss_rewrite_relative_paths',
      'views_rss_htmlspecialchars',
    'help' => 'http://www.rssboard.org/rss-profile#element-channel-description',
  );
  $elements['link'] = array(
    'configurable' => FALSE,
    'preprocess functions' => array('views_rss_core_preprocess_channel_link'),
    'help' => 'http://www.rssboard.org/rss-profile#element-channel-link',
  $elements['atom:link'] = array(
    'configurable' => FALSE,
    'preprocess functions' => array('views_rss_core_preprocess_channel_atom_link'),
    'help' => 'http://www.rssboard.org/rss-profile#namespace-elements-atom-link',
  );
    'description' => t('The language the channel is written in. This allows aggregators to group all Italian language sites, for example, on a single page. See list of <a href="@w3c_url">allowable values</a> for this element defined by the W3C.', array(
      '@w3c_url' => url('http://www.w3.org/TR/REC-html40/struct/dirlang.html', array('fragment' => 'langcodes')),
    )),
    'help' => 'http://www.rssboard.org/rss-profile#element-channel-language',
  $elements['category'] = array(
    'description' => t('Specify one or more categories that the channel belongs to. Separate multiple items with comma.'),
    'preprocess functions' => array(
      'views_rss_htmlspecialchars',
      'views_rss_core_preprocess_channel_category',
    'help' => 'http://www.rssboard.org/rss-profile#element-channel-category',
  );
  $elements['image'] = array(
    'description' => t('Path to the image to be used as the artwork for your feed, for example <em>sites/default/files/AllAboutEverything.jpg</em>. Allowed image formats are GIF, JPEG or PNG. Maximum image width is 144 pixels, maximum height 400 pixels.'),
    'preprocess functions' => array('views_rss_core_preprocess_channel_image'),
    'help' => 'http://www.rssboard.org/rss-profile#element-channel-image',
  );
    'description' => t('Copyright notice for content in the channel.'),
    'preprocess functions' => array('views_rss_htmlspecialchars'),
    'help' => 'http://www.rssboard.org/rss-profile#element-channel-copyright',
    'description' => t('Email address for person responsible for editorial content.'),
    'preprocess functions' => array('views_rss_htmlspecialchars'),
    'help' => 'http://www.rssboard.org/rss-profile#element-channel-managingeditor',
    'description' => t('Email address for person responsible for technical issues relating to channel.'),
    'preprocess functions' => array('views_rss_htmlspecialchars'),
    'help' => 'http://www.rssboard.org/rss-profile#element-channel-webmaster',
    'description' => t('A string indicating the program used to generate the channel.'),
    'preprocess functions' => array('views_rss_htmlspecialchars'),
    'help' => 'http://www.rssboard.org/rss-profile#element-channel-generator',
    'description' => t("A URL that points to the documentation for the format used in the RSS file. It's for people who might stumble across an RSS file on a Web server 25 years from now and wonder what it is."),
    'help' => 'http://www.rssboard.org/rss-profile#element-channel-docs',
  );
  $elements['cloud'] = array(
    'description' => t("Allows processes to register with a cloud to be notified of updates to the channel, implementing a lightweight publish-subscribe protocol for RSS feeds. Example: <em>soap://rpc.sys.com:80/RPC2#pingMe</em>"),
    'preprocess functions' => array('views_rss_core_preprocess_channel_cloud'),
    'help' => 'http://www.rssboard.org/rss-profile#element-channel-cloud',
    'description' => t("ttl stands for time to live. It's a number of minutes that indicates how long a channel can be cached before refreshing from the source."),
    'help' => 'http://www.rssboard.org/rss-profile#element-channel-ttl',
    'description' => t('A hint for aggregators telling them which hours they can skip. The hours must be expressed as an integer representing the number of hours since 00:00:00 GMT. Values from 0 to 23 are permitted, with 0 representing midnight. An hour must not be duplicated.'),
    'preprocess functions' => array('views_rss_core_preprocess_channel_skip'),
    'help' => 'http://www.rssboard.org/rss-profile#element-channel-skiphours',
    'description' => t('A hint for aggregators telling them which days of the week they can skip (up to seven days).'),
    'preprocess functions' => array('views_rss_core_preprocess_channel_skip'),
    'help' => 'http://www.rssboard.org/rss-profile#element-channel-skipdays',
  $elements['pubDate'] = array(
    'configurable' => FALSE,
    'preprocess functions' => array('views_rss_core_preprocess_channel_date'),
    'help' => 'http://www.rssboard.org/rss-profile#element-channel-pubdate',
  );
  $elements['lastBuildDate'] = array(
    'configurable' => FALSE,
    'preprocess functions' => array('views_rss_core_preprocess_channel_date'),
    'help' => 'http://www.rssboard.org/rss-profile#element-channel-lastbuilddate',
  );
 * Implementation of hook_views_rss_item_elements().
function views_rss_core_views_rss_item_elements() {
    'description' => t('The title of the item. Required by RSS specification.'),
    'preprocess functions' => array('views_rss_htmlspecialchars'),
    'help' => 'http://www.rssboard.org/rss-profile#element-channel-item-title',
    'description' => t('The URL of the item. Required by RSS specification.'),
    'help' => 'http://www.rssboard.org/rss-profile#element-channel-item-link',
    'description' => t('The item synopsis. Required by RSS specification.'),
    'preprocess functions' => array(
      'views_rss_rewrite_relative_paths',
    'help' => 'http://www.rssboard.org/rss-profile#element-channel-item-description',
    'description' => t('Email address of the author of the item.'),
    'help' => 'http://www.rssboard.org/rss-profile#element-channel-item-author',
    'description' => t('Includes the item in one or more categories.'),
    'help' => 'http://www.rssboard.org/rss-profile#element-channel-item-category',
    'description' => t('URL of a page for comments relating to the item.'),
    'help' => 'http://www.rssboard.org/rss-profile#element-channel-item-comments',
    'description' => t('Describes a media object that is attached to the item.'),
    'help' => 'http://www.rssboard.org/rss-profile#element-channel-item-enclosure',
    'description' => t('A string that uniquely identifies the item.'),
    'preprocess functions' => array('views_rss_core_preprocess_item_guid'),
    'help' => 'http://www.rssboard.org/rss-profile#element-channel-item-guid',
    'description' => t('Indicates when the item was published.'),
    'help' => 'http://www.rssboard.org/rss-profile#element-channel-item-pubdate',
  );
  $elements['source'] = array(
    'configurable' => FALSE,
    'description' => t('The RSS channel that the item came from.'),
    'preprocess functions' => array('views_rss_core_preprocess_item_source'),
    'help' => 'http://www.rssboard.org/rss-profile#element-channel-item-source',
 * Implementation of hook_views_rss_date_sources().
function views_rss_core_views_rss_date_sources() {
  $sources['node'] = array(
    'pubDate' => array(
  $sources['node_revisions'] = array(
    'pubDate' => array(
      'table' => 'node_revisions',
    ),
  );
  $sources['files'] = array(
    'pubDate' => array(
      'table' => 'files',
    ),
  );
  $sources['users'] = array(
    'pubDate' => array(
      'table' => 'users',
 * Implementation of hook_views_query_alter().
function views_rss_core_views_query_alter(&$view, &$query) {
  if ($view->style_plugin->definition['handler'] == 'views_rss_plugin_style_fields') {
    foreach (array_keys($date_sources) as $module) {
      if (isset($date_sources[$module][$view->base_table])) {
        // Select the most recent node creation date for <pubDate> element.
        if (isset($date_sources[$module][$view->base_table]['pubDate'])) {
          $query->fields['pubDate'] = $date_sources[$module][$view->base_table]['pubDate'];
        }
        // Select the most recent node update date for <lastBuildDate> element.
        if (isset($date_sources[$module][$view->base_table]['lastBuildDate'])) {
          $query->fields['lastBuildDate'] = $date_sources[$module][$view->base_table]['lastBuildDate'];
        }
      }
    }
  }
}

/**
 * Implementation of hook_theme().
 */
function views_rss_core_theme() {
  $theme = array(
    // CCK field formatters.
    'views_rss_core_formatter_enclosure_image' => array(
      'arguments' => array('element' => NULL),
      'function' => 'views_rss_core_field_formatter_view',
      'file' => 'views_rss_core.field.inc',
    ),
    'views_rss_core_formatter_enclosure_file' => array(
      'arguments' => array('element' => NULL),
      'function' => 'views_rss_core_field_formatter_view',
      'file' => 'views_rss_core.field.inc',
    ),
  );
  if (module_exists('imagecache')) {
    foreach (imagecache_presets() as $preset) {
      $theme['views_rss_core_formatter_enclosure_image' . $preset['presetname']] = array(
        'arguments' => array('element' => NULL),
        'function' => 'views_rss_core_field_formatter_view',
        'file' => 'views_rss_core.field.inc',
/**
 * Implements hook_views_rss_options_form_validate().
 */
function views_rss_core_views_rss_options_form_validate($form, &$form_state) {

  // Validate channel <image> element.
  if (!empty($form_state['values']['style_options']['channel']['core']['views_rss_core']['image'])) {
    // Do not validate absolute URLs, as this could mean external image.
    if (!valid_url($form_state['values']['style_options']['channel']['core']['views_rss_core']['image'], TRUE)) {
      // Check that image exists and is in acceptable format.
      $real_path = realpath($form_state['values']['style_options']['channel']['core']['views_rss_core']['image']);
      if (!file_exists($real_path)) {
        form_set_error('style_options][channel][core][views_rss_core][image', t('Unable to find %image or incorrect image format.', array(
          '%image' => $form_state['values']['style_options']['channel']['core']['views_rss_core']['image'],
        )));
      }
      else {
        list($width, $height) = getimagesize($real_path);
        // Check image width.
        if ($width > 144) {
          form_set_error('style_options][channel][core][views_rss_core][image', t("Maximum allowed width of an image for feed channel's &lt;image&gt; element is 144 pixels. Specified %image is !width pixels wide.", array(
            '%image' => $form_state['values']['style_options']['channel']['core']['views_rss_core']['image'],
            '!width' => $width,
          )));
        }
        // Check image height.
        if ($height > 400) {
          form_set_error('style_options][channel][core][views_rss_core][image', t("Maximum allowed height of an image for feed channel's &lt;image&gt; element is 400 pixels. Specified %image is !height pixels high.", array(
            '%image' => $form_state['values']['style_options']['channel']['core']['views_rss_core']['image'],
            '!height' => $height,
          )));
        }

  // Validate channel <docs> element.
  if (!empty($form_state['values']['style_options']['channel']['core']['views_rss_core']['docs'])) {
    if (!valid_url($form_state['values']['style_options']['channel']['core']['views_rss_core']['docs'], TRUE)) {
      form_set_error('style_options][channel][core][views_rss_core][docs', t("Not a valid URL."));
    }
  }
}