From e4c24df2f3a2853f57dfe6828b1331746b011fd8 Mon Sep 17 00:00:00 2001 From: Dries <dries@buytaert.net> Date: Wed, 19 Jun 2013 16:29:36 -0400 Subject: [PATCH] Issue #1821844 by dawehner, jibran, damiankloip, xjm, ParisLiakos, wamilton, olli: Aggregator views integration. --- core/modules/aggregator/aggregator.views.inc | 324 ++++++++++++++++++ .../aggregator/Plugin/Core/Entity/Item.php | 4 +- .../Plugin/views/argument/CategoryCid.php | 67 ++++ .../aggregator/Plugin/views/argument/Fid.php | 69 ++++ .../aggregator/Plugin/views/argument/Iid.php | 69 ++++ .../Plugin/views/field/Category.php | 84 +++++ .../Plugin/views/field/TitleLink.php | 88 +++++ .../aggregator/Plugin/views/field/Xss.php | 29 ++ .../Plugin/views/filter/CategoryCid.php | 65 ++++ .../aggregator/Plugin/views/row/Rss.php | 109 ++++++ .../Tests/Views/IntegrationTest.php | 114 ++++++ .../aggregator_test_views.info.yml | 9 + .../aggregator_test_views.module | 1 + .../views.view.test_aggregator_items.yml | 168 +++++++++ 14 files changed, 1199 insertions(+), 1 deletion(-) create mode 100644 core/modules/aggregator/aggregator.views.inc create mode 100644 core/modules/aggregator/lib/Drupal/aggregator/Plugin/views/argument/CategoryCid.php create mode 100644 core/modules/aggregator/lib/Drupal/aggregator/Plugin/views/argument/Fid.php create mode 100644 core/modules/aggregator/lib/Drupal/aggregator/Plugin/views/argument/Iid.php create mode 100644 core/modules/aggregator/lib/Drupal/aggregator/Plugin/views/field/Category.php create mode 100644 core/modules/aggregator/lib/Drupal/aggregator/Plugin/views/field/TitleLink.php create mode 100644 core/modules/aggregator/lib/Drupal/aggregator/Plugin/views/field/Xss.php create mode 100644 core/modules/aggregator/lib/Drupal/aggregator/Plugin/views/filter/CategoryCid.php create mode 100644 core/modules/aggregator/lib/Drupal/aggregator/Plugin/views/row/Rss.php create mode 100644 core/modules/aggregator/lib/Drupal/aggregator/Tests/Views/IntegrationTest.php create mode 100644 core/modules/aggregator/tests/modules/aggregator_test_views/aggregator_test_views.info.yml create mode 100644 core/modules/aggregator/tests/modules/aggregator_test_views/aggregator_test_views.module create mode 100755 core/modules/aggregator/tests/modules/aggregator_test_views/test_views/views.view.test_aggregator_items.yml diff --git a/core/modules/aggregator/aggregator.views.inc b/core/modules/aggregator/aggregator.views.inc new file mode 100644 index 000000000000..5d87034496cf --- /dev/null +++ b/core/modules/aggregator/aggregator.views.inc @@ -0,0 +1,324 @@ +<?php + +/** + * @file + * Provides views data for aggregator.module. + * + * @ingroup views_module_handlers + */ + +/** + * Implements hook_views_data(). + */ +function aggregator_views_data() { + $data = array(); + + $data['aggregator_item']['table']['group'] = t('Aggregator'); + + $data['aggregator_item']['table']['base'] = array( + 'field' => 'iid', + 'title' => t('Aggregator item'), + 'help' => t('Aggregator items are imported from external RSS and Atom news feeds.'), + ); + $data['aggregator_item']['table']['entity type'] = 'aggregator_item'; + + $data['aggregator_item']['iid'] = array( + 'title' => t('Item ID'), + 'help' => t('The unique ID of the aggregator item.'), + 'field' => array( + 'id' => 'numeric', + ), + 'argument' => array( + 'id' => 'aggregator_iid', + 'name field' => 'title', + 'numeric' => TRUE, + ), + 'filter' => array( + 'id' => 'numeric', + ), + 'sort' => array( + 'id' => 'standard', + ), + ); + + $data['aggregator_item']['title'] = array( + 'title' => t('Title'), + 'help' => t('The title of the aggregator item.'), + 'field' => array( + 'id' => 'aggregator_title_link', + 'extra' => array('link'), + ), + 'argument' => array( + 'id' => 'string', + ), + 'sort' => array( + 'id' => 'standard', + ), + 'filter' => array( + 'id' => 'string', + ), + ); + + $data['aggregator_item']['link'] = array( + 'title' => t('Link'), + 'help' => t('The link to the original source URL of the item.'), + 'field' => array( + 'id' => 'url', + ), + 'argument' => array( + 'id' => 'string', + ), + 'sort' => array( + 'id' => 'standard', + ), + 'filter' => array( + 'id' => 'string', + ), + ); + + $data['aggregator_item']['author'] = array( + 'title' => t('Author'), + 'help' => t('The author of the original imported item.'), + 'field' => array( + 'id' => 'aggregator_xss', + ), + 'argument' => array( + 'id' => 'string', + ), + 'sort' => array( + 'id' => 'standard', + ), + 'filter' => array( + 'id' => 'string', + ), + ); + + $data['aggregator_item']['guid'] = array( + 'title' => t('GUID'), + 'help' => t('The guid of the original imported item.'), + 'field' => array( + 'id' => 'standard', + ), + 'argument' => array( + 'id' => 'string', + ), + 'sort' => array( + 'id' => 'standard', + ), + 'filter' => array( + 'id' => 'string', + ), + ); + + $data['aggregator_item']['description'] = array( + 'title' => t('Body'), + 'help' => t('The actual content of the imported item.'), + 'field' => array( + 'id' => 'aggregator_xss', + 'click sortable' => FALSE, + ), + 'argument' => array( + 'id' => 'string', + ), + 'filter' => array( + 'id' => 'string', + ), + 'sort' => array( + 'id' => 'standard', + ), + ); + + $data['aggregator_item']['timestamp'] = array( + 'title' => t('Timestamp'), + 'help' => t('The date the original feed item was posted. (With some feeds, this will be the date it was imported.)'), + 'field' => array( + 'id' => 'date', + ), + 'sort' => array( + 'id' => 'date', + ), + 'filter' => array( + 'id' => 'date', + ), + 'argument' => array( + 'id' => 'date', + ), + ); + + $data['aggregator_feed']['table']['group'] = t('Aggregator feed'); + + $data['aggregator_feed']['table']['base'] = array( + 'field' => 'fid', + 'title' => t('Aggregator feed'), + ); + + $data['aggregator_feed']['table']['entity type'] = 'aggregator_feed'; + + $data['aggregator_feed']['table']['join'] = array( + 'aggregator_item' => array( + 'left_field' => 'fid', + 'field' => 'fid', + ), + ); + + $data['aggregator_feed']['fid'] = array( + 'title' => t('Feed ID'), + 'help' => t('The unique ID of the aggregator feed.'), + 'field' => array( + 'id' => 'numeric', + ), + 'argument' => array( + 'id' => 'aggregator_fid', + 'name field' => 'title', + 'numeric' => TRUE, + ), + 'filter' => array( + 'id' => 'numeric', + ), + 'sort' => array( + 'id' => 'standard', + ), + ); + + $data['aggregator_feed']['title'] = array( + 'title' => t('Title'), + 'help' => t('The title of the aggregator feed.'), + 'field' => array( + 'id' => 'aggregator_title_link', + 'extra' => array('link'), + ), + 'sort' => array( + 'id' => 'standard', + ), + 'filter' => array( + 'id' => 'string', + ), + 'argument' => array( + 'id' => 'string', + ), + ); + + $data['aggregator_feed']['link'] = array( + 'title' => t('Link'), + 'help' => t('The link to the source URL of the feed.'), + 'field' => array( + 'id' => 'url', + ), + 'argument' => array( + 'id' => 'string', + ), + 'sort' => array( + 'id' => 'standard', + ), + 'filter' => array( + 'id' => 'string', + ), + ); + + $data['aggregator_feed']['checked'] = array( + 'title' => t('Last checked'), + 'help' => t('The date the feed was last checked for new content.'), + 'field' => array( + 'id' => 'date', + ), + 'sort' => array( + 'id' => 'date', + ), + 'filter' => array( + 'id' => 'date', + ), + 'argument' => array( + 'id' => 'date', + ), + ); + + $data['aggregator_feed']['description'] = array( + 'title' => t('Description'), + 'help' => t('The description of the aggregator feed.'), + 'field' => array( + 'id' => 'xss', + 'click sortable' => FALSE, + ), + 'filter' => array( + 'id' => 'string', + ), + 'argument' => array( + 'id' => 'string', + ), + 'sort' => array( + 'id' => 'standard', + ), + ); + + $data['aggregator_feed']['modified'] = array( + 'title' => t('Last modified'), + 'help' => t('The date of the most recent new content on the feed.'), + 'field' => array( + 'id' => 'date', + ), + 'sort' => array( + 'id' => 'date', + ), + 'filter' => array( + 'id' => 'date', + ), + 'argument' => array( + 'id' => 'date', + ), + ); + + $data['aggregator_category_feed']['table']['join'] = array( + 'aggregator_item' => array( + 'left_field' => 'fid', + 'field' => 'fid', + ), + ); + + $data['aggregator_category']['table']['group'] = t('Aggregator category'); + + $data['aggregator_category']['table']['join'] = array( + 'aggregator_item' => array( + 'left_table' => 'aggregator_category_feed', + 'left_field' => 'cid', + 'field' => 'cid', + ), + ); + + $data['aggregator_category']['cid'] = array( + 'title' => t('Category ID'), + 'help' => t('The unique ID of the aggregator category.'), + 'field' => array( + 'id' => 'numeric', + ), + 'argument' => array( + 'id' => 'aggregator_category_cid', + 'name field' => 'title', + 'numeric' => TRUE, + ), + 'filter' => array( + 'id' => 'aggregator_category_cid', + ), + 'sort' => array( + 'id' => 'standard', + ), + ); + + $data['aggregator_category']['title'] = array( + 'title' => t('Category'), + 'help' => t('The title of the aggregator category.'), + 'field' => array( + 'id' => 'aggregator_category', + ), + 'argument' => array( + 'id' => 'string', + ), + 'sort' => array( + 'id' => 'standard', + ), + 'filter' => array( + 'id' => 'string', + ), + ); + + return $data; +} diff --git a/core/modules/aggregator/lib/Drupal/aggregator/Plugin/Core/Entity/Item.php b/core/modules/aggregator/lib/Drupal/aggregator/Plugin/Core/Entity/Item.php index b5277525f413..6da20f30bcad 100644 --- a/core/modules/aggregator/lib/Drupal/aggregator/Plugin/Core/Entity/Item.php +++ b/core/modules/aggregator/lib/Drupal/aggregator/Plugin/Core/Entity/Item.php @@ -136,7 +136,9 @@ public function label($langcode = NULL) { * {@inheritdoc} */ public function postCreate(EntityStorageControllerInterface $storage_controller) { - $this->timestamp->value = REQUEST_TIME; + if (!isset($this->timestamp->value)) { + $this->timestamp->value = REQUEST_TIME; + } } /** diff --git a/core/modules/aggregator/lib/Drupal/aggregator/Plugin/views/argument/CategoryCid.php b/core/modules/aggregator/lib/Drupal/aggregator/Plugin/views/argument/CategoryCid.php new file mode 100644 index 000000000000..fb55a405a38e --- /dev/null +++ b/core/modules/aggregator/lib/Drupal/aggregator/Plugin/views/argument/CategoryCid.php @@ -0,0 +1,67 @@ +<?php + +/** + * @file + * Contains \Drupal\aggregator\Plugin\views\argument\CategoryCid. + */ + +namespace Drupal\aggregator\Plugin\views\argument; + +use Drupal\views\Plugin\views\argument\Numeric; +use Drupal\Component\Annotation\PluginID; +use Drupal\Component\Utility\String; +use Drupal\Core\Database\Connection; +use Symfony\Component\DependencyInjection\ContainerInterface; + +/** + * Argument handler to accept an aggregator category id. + * + * @ingroup views_argument_handlers + * + * @PluginID("aggregator_category_cid") + */ +class CategoryCid extends Numeric { + + /** + * Database Service Object. + * + * @var \Drupal\Core\Database\Connection + */ + protected $database; + + /** + * Constructs a Drupal\Component\Plugin\PluginBase object. + * + * @param array $configuration + * A configuration array containing information about the plugin instance. + * @param string $plugin_id + * The plugin_id for the plugin instance. + * @param array $plugin_definition + * The plugin implementation definition. + * @param \Drupal\Core\Database\Connection $database + * Database Service Object. + */ + public function __construct(array $configuration, $plugin_id, array $plugin_definition, Connection $database) { + parent::__construct($configuration, $plugin_id, $plugin_definition); + $this->database = $database; + } + + /** + * {@inheritdoc} + */ + public static function create(ContainerInterface $container, array $configuration, $plugin_id, array $plugin_definition) { + return new static($configuration, $plugin_id, $plugin_definition, $container->get('database')); + } + + /** + * {@inheritdoc} + */ + function titleQuery() { + $titles = $this->database->query("SELECT title FROM {aggregator_category} where cid IN (:cid)", array(':cid' => $this->value))->fetchCol(); + + return array_map(function ($title) { + return String::checkPlain($title); + }, $titles); + } + +} diff --git a/core/modules/aggregator/lib/Drupal/aggregator/Plugin/views/argument/Fid.php b/core/modules/aggregator/lib/Drupal/aggregator/Plugin/views/argument/Fid.php new file mode 100644 index 000000000000..f77fb02fbe7c --- /dev/null +++ b/core/modules/aggregator/lib/Drupal/aggregator/Plugin/views/argument/Fid.php @@ -0,0 +1,69 @@ +<?php + +/** + * @file + * Contains \Drupal\aggregator\Plugin\views\argument\Fid. + */ + +namespace Drupal\aggregator\Plugin\views\argument; + +use Drupal\views\Plugin\views\argument\Numeric; +use Drupal\Component\Annotation\PluginID; +use Drupal\Component\Utility\String; +use Drupal\Core\Entity\EntityManager; +use Symfony\Component\DependencyInjection\ContainerInterface; + +/** + * Argument handler to accept an aggregator feed id. + * + * @ingroup views_argument_handlers + * + * @PluginID("aggregator_fid") + */ +class Fid extends Numeric { + + /** + * The entity manager service + * + * @var \Drupal\Core\Entity\EntityManager + */ + protected $entityManager; + + /** + * Constructs a Drupal\Component\Plugin\PluginBase object. + * + * @param array $configuration + * A configuration array containing information about the plugin instance. + * @param string $plugin_id + * The plugin_id for the plugin instance. + * @param array $plugin_definition + * The plugin implementation definition. + * @param \Drupal\Core\Entity\EntityManager $entity_manager + * The entity manager. + */ + public function __construct(array $configuration, $plugin_id, array $plugin_definition, EntityManager $entity_manager) { + parent::__construct($configuration, $plugin_id, $plugin_definition); + $this->entityManager = $entity_manager; + } + + /** + * {@inheritdoc} + */ + public static function create(ContainerInterface $container, array $configuration, $plugin_id, array $plugin_definition) { + return new static($configuration, $plugin_id, $plugin_definition, $container->get('plugin.manager.entity')); + } + + /** + * {@inheritdoc} + */ + function titleQuery() { + $titles = array(); + + $feeds = $this->entityManager->getStorageController('aggregator_feed')->load($this->value); + foreach ($feeds as $feed) { + $titles[] = String::checkPlain($feed->label()); + } + return $titles; + } + +} diff --git a/core/modules/aggregator/lib/Drupal/aggregator/Plugin/views/argument/Iid.php b/core/modules/aggregator/lib/Drupal/aggregator/Plugin/views/argument/Iid.php new file mode 100644 index 000000000000..1ce18643814a --- /dev/null +++ b/core/modules/aggregator/lib/Drupal/aggregator/Plugin/views/argument/Iid.php @@ -0,0 +1,69 @@ +<?php + +/** + * @file + * Contains \Drupal\aggregator\Plugin\views\argument\Iid. + */ + +namespace Drupal\aggregator\Plugin\views\argument; + +use Drupal\views\Plugin\views\argument\Numeric; +use Drupal\Component\Annotation\PluginID; +use Drupal\Component\Utility\String; +use Drupal\Core\Entity\EntityManager; +use Symfony\Component\DependencyInjection\ContainerInterface; + +/** + * Argument handler to accept an aggregator item id. + * + * @ingroup views_argument_handlers + * + * @PluginID("aggregator_iid") + */ +class Iid extends Numeric { + + /** + * The entity manager service + * + * @var \Drupal\Core\Entity\EntityManager + */ + protected $entityManager; + + /** + * Constructs a Drupal\Component\Plugin\PluginBase object. + * + * @param array $configuration + * A configuration array containing information about the plugin instance. + * @param string $plugin_id + * The plugin_id for the plugin instance. + * @param array $plugin_definition + * The plugin implementation definition. + * @param \Drupal\Core\Entity\EntityManager $entity_manager + * The entity manager. + */ + public function __construct(array $configuration, $plugin_id, array $plugin_definition, EntityManager $entity_manager) { + parent::__construct($configuration, $plugin_id, $plugin_definition); + $this->entityManager = $entity_manager; + } + + /** + * {@inheritdoc} + */ + public static function create(ContainerInterface $container, array $configuration, $plugin_id, array $plugin_definition) { + return new static($configuration, $plugin_id, $plugin_definition, $container->get('plugin.manager.entity')); + } + + /** + * {@inheritdoc} + */ + function titleQuery() { + $titles = array(); + + $items = $this->entityManager->getStorageController('aggregator_item')->load($this->value); + foreach ($items as $feed) { + $titles[] = String::checkPlain($feed->label()); + } + return $titles; + } + +} diff --git a/core/modules/aggregator/lib/Drupal/aggregator/Plugin/views/field/Category.php b/core/modules/aggregator/lib/Drupal/aggregator/Plugin/views/field/Category.php new file mode 100644 index 000000000000..829142294cb6 --- /dev/null +++ b/core/modules/aggregator/lib/Drupal/aggregator/Plugin/views/field/Category.php @@ -0,0 +1,84 @@ +<?php + +/** + * @file + * Contains \Drupal\aggregator\Plugin\views\field\Category. + */ + +namespace Drupal\aggregator\Plugin\views\field; + +use Drupal\views\Plugin\views\display\DisplayPluginBase; +use Drupal\views\Plugin\views\field\FieldPluginBase; +use Drupal\views\ViewExecutable; +use Drupal\Component\Annotation\PluginID; + +/** + * Defines a simple renderer that allows linking to an aggregator category. + * + * @ingroup views_field_handlers + * + * @PluginID("aggregator_category") + */ +class Category extends FieldPluginBase { + + /** + * {@inheritdoc} + */ + public function init(ViewExecutable $view, DisplayPluginBase $display, array &$options = NULL) { + parent::init($view, $display, $options); + + $this->additional_fields['cid'] = 'cid'; + } + + /** + * {@inheritdoc} + */ + protected function defineOptions() { + $options = parent::defineOptions(); + + $options['link_to_category'] = array('default' => FALSE); + return $options; + } + + /** + * {@inheritdoc} + */ + public function buildOptionsForm(&$form, &$form_state) { + $form['link_to_category'] = array( + '#title' => t('Link this field to its aggregator category page'), + '#description' => t('This will override any other link you have set.'), + '#type' => 'checkbox', + '#default_value' => !empty($this->options['link_to_category']), + ); + parent::buildOptionsForm($form, $form_state); + } + + /** + * Render whatever the data is as a link to the category. + * + * @param string $data + * The XSS safe string for the link text. + * @param object $values + * The values retrieved from the database. + * + * @return data + * Returns string for the link text. + */ + protected function render_link($data, $values) { + $cid = $this->getValue($values, 'cid'); + if (!empty($this->options['link_to_category']) && !empty($cid) && $data !== NULL && $data !== '') { + $this->options['alter']['make_link'] = TRUE; + $this->options['alter']['path'] = "aggregator/categories/$cid"; + } + return $data; + } + + /** + * {@inheritdoc} + */ + public function render($values) { + $value = $this->getValue($values); + return $this->render_link($this->sanitizeValue($value), $values); + } + +} diff --git a/core/modules/aggregator/lib/Drupal/aggregator/Plugin/views/field/TitleLink.php b/core/modules/aggregator/lib/Drupal/aggregator/Plugin/views/field/TitleLink.php new file mode 100644 index 000000000000..3eae3778ec6c --- /dev/null +++ b/core/modules/aggregator/lib/Drupal/aggregator/Plugin/views/field/TitleLink.php @@ -0,0 +1,88 @@ +<?php + +/** + * @file + * Contains \Drupal\aggregator\Plugin\views\field\TitleLink. + */ + +namespace Drupal\aggregator\Plugin\views\field; + +use Drupal\views\Plugin\views\display\DisplayPluginBase; +use Drupal\views\Plugin\views\field\FieldPluginBase; +use Drupal\views\ViewExecutable; +use Drupal\Component\Annotation\PluginID; + +/** + * Defines a field handler that turns an item's title into a clickable link to + * the original source article. + * + * @ingroup views_field_handlers + * + * @PluginID("aggregator_title_link") + */ +class TitleLink extends FieldPluginBase { + + /** + * {@inheritdoc} + */ + public function init(ViewExecutable $view, DisplayPluginBase $display, array &$options = NULL) { + parent::init($view, $display, $options); + + $this->additional_fields['link'] = 'link'; + } + + /** + * {@inheritdoc} + */ + protected function defineOptions() { + $options = parent::defineOptions(); + + $options['display_as_link'] = array('default' => TRUE); + + return $options; + } + + /** + * {@inheritdoc} + */ + public function buildOptionsForm(&$form, &$form_state) { + $form['display_as_link'] = array( + '#title' => t('Display as link'), + '#type' => 'checkbox', + '#default_value' => !empty($this->options['display_as_link']), + ); + parent::buildOptionsForm($form, $form_state); + } + + /** + * {@inheritdoc} + */ + function render($values) { + $value = $this->getValue($values); + return $this->render_link($this->sanitizeValue($value), $values); + } + + /** + * Renders aggregator item's title as link. + * + * @param string $data + * The XSS safe string for the link text. + * @param object $values + * The values retrieved from the database. + * + * @return data + * Returns string for the link text. + */ + protected function render_link($data, $values) { + $link = $this->getValue($values, 'link'); + if (!empty($this->options['display_as_link'])) { + $this->options['alter']['make_link'] = TRUE; + $this->options['alter']['path'] = $link; + $this->options['alter']['html'] = TRUE; + $this->options['alter']['absolute'] = TRUE; + } + + return $data; + } + +} diff --git a/core/modules/aggregator/lib/Drupal/aggregator/Plugin/views/field/Xss.php b/core/modules/aggregator/lib/Drupal/aggregator/Plugin/views/field/Xss.php new file mode 100644 index 000000000000..5b755ffec94b --- /dev/null +++ b/core/modules/aggregator/lib/Drupal/aggregator/Plugin/views/field/Xss.php @@ -0,0 +1,29 @@ +<?php + +/** + * @file + * Contains \Drupal\aggregator\Plugin\views\field\Xss. + */ + +namespace Drupal\aggregator\Plugin\views\field; + +use Drupal\views\Plugin\views\field\Xss as XssBase; +use Drupal\Component\Annotation\PluginID; + +/** + * Filters htmls tags from item. + * + * @ingroup views_field_handlers + * + * @PluginID("aggregator_xss") + */ +class Xss extends XssBase { + + /** + * {@inheritdoc} + */ + public function sanitizeValue($value, $type = NULL) { + return aggregator_filter_xss($value); + } + +} diff --git a/core/modules/aggregator/lib/Drupal/aggregator/Plugin/views/filter/CategoryCid.php b/core/modules/aggregator/lib/Drupal/aggregator/Plugin/views/filter/CategoryCid.php new file mode 100644 index 000000000000..bb5f70555248 --- /dev/null +++ b/core/modules/aggregator/lib/Drupal/aggregator/Plugin/views/filter/CategoryCid.php @@ -0,0 +1,65 @@ +<?php + +/** + * @file + * Contains \Drupal\aggregator\Plugin\views\filter\CategoryCid. + */ + +namespace Drupal\aggregator\Plugin\views\filter; + +use Drupal\views\Plugin\views\filter\InOperator; +use Drupal\Component\Annotation\PluginID; + +/** + * Defines a filter handler that filters by aggregator category cid. + * + * @ingroup views_filter_handlers + * + * @PluginID("aggregator_category_cid") + */ +class CategoryCid extends InOperator { + + /** + * Database Service Object. + * + * @var \Drupal\Core\Database\Connection + */ + protected $database; + + /** + * Constructs a Drupal\Component\Plugin\PluginBase object. + * + * @param array $configuration + * A configuration array containing information about the plugin instance. + * @param string $plugin_id + * The plugin_id for the plugin instance. + * @param array $plugin_definition + * The plugin implementation definition. + * @param \Drupal\Core\Database\Connection $database + * Database Service Object. + */ + public function __construct(array $configuration, $plugin_id, array $plugin_definition, Connection $database) { + parent::__construct($configuration, $plugin_id, $plugin_definition); + $this->database = $database; + } + + /** + * {@inheritdoc} + */ + public static function create(ContainerInterface $container, array $configuration, $plugin_id, array $plugin_definition) { + return new static($configuration, $plugin_id, $plugin_definition, $container->get('database')); + } + + /** + * {@inheritdoc} + */ + function getValueOptions() { + if (isset($this->value_options)) { + return; + } + + $this->value_options = array(); + $this->value_options = $this->database->query('SELECT cid, title FROM {aggregator_category} ORDER BY title')->fetchAllKeyed(); + } + +} diff --git a/core/modules/aggregator/lib/Drupal/aggregator/Plugin/views/row/Rss.php b/core/modules/aggregator/lib/Drupal/aggregator/Plugin/views/row/Rss.php new file mode 100644 index 000000000000..7e55034e53a4 --- /dev/null +++ b/core/modules/aggregator/lib/Drupal/aggregator/Plugin/views/row/Rss.php @@ -0,0 +1,109 @@ +<?php + +/** + * @file + * Contains \Drupal\aggregator\Plugin\views\row\Rss. + */ + +namespace Drupal\aggregator\Plugin\views\row; + +use Drupal\views\Plugin\views\row\RowPluginBase; +use Drupal\Component\Annotation\Plugin; +use Drupal\Core\Annotation\Translation; + +/** + * Defines a row plugin which loads an aggregator item and renders as RSS. + * + * @Plugin( + * id = "aggregator_rss", + * module = "aggregator", + * theme = "views_view_row_rss", + * title = @Translation("Aggregator item"), + * help = @Translation("Display the aggregator item using the data from the original source."), + * base = {"aggregator_item"}, + * display_types = {"feed"} + * ) + */ +class Rss extends RowPluginBase { + + /** + * The table the aggregator item is using for storage. + * + * @var string + */ + public $base_table = 'aggregator_item'; + + /** + * The actual field which is used to identify a aggregator item. + * + * @var string + */ + public $base_field = 'iid'; + + /** + * {@inheritdoc} + */ + protected function defineOptions() { + $options = parent::defineOptions(); + + $options['item_length'] = array('default' => 'default'); + + return $options; + } + + /** + * {@inheritdoc} + */ + public function buildOptionsForm(&$form, &$form_state) { + $form['item_length'] = array( + '#type' => 'select', + '#title' => t('Display type'), + '#options' => array( + 'fulltext' => t('Full text'), + 'teaser' => t('Title plus teaser'), + 'title' => t('Title only'), + 'default' => t('Use default RSS settings'), + ), + '#default_value' => $this->options['item_length'], + ); + } + + /** + * {@inheritdoc} + */ + function render($row) { + $entity = $row->_entity; + + $item = new \stdClass(); + foreach ($entity->getProperties() as $name => $value) { + // views_view_row_rss takes care about the escaping. + $item->{$name} = $value->value; + } + + $item->elements = array( + array( + 'key' => 'pubDate', + // views_view_row_rss takes care about the escaping. + 'value' => gmdate('r', $entity->timestamp->value), + ), + array( + 'key' => 'dc:creator', + // views_view_row_rss takes care about the escaping. + 'value' => $entity->author->value, + ), + array( + 'key' => 'guid', + // views_view_row_rss takes care about the escaping. + 'value' => $entity->guid->value, + 'attributes' => array('isPermaLink' => 'false'), + ), + ); + + return theme($this->themeFunctions(), array( + 'view' => $this->view, + 'options' => $this->options, + 'row' => $item, + )); + } + +} diff --git a/core/modules/aggregator/lib/Drupal/aggregator/Tests/Views/IntegrationTest.php b/core/modules/aggregator/lib/Drupal/aggregator/Tests/Views/IntegrationTest.php new file mode 100644 index 000000000000..da85242ae6ce --- /dev/null +++ b/core/modules/aggregator/lib/Drupal/aggregator/Tests/Views/IntegrationTest.php @@ -0,0 +1,114 @@ +<?php + +/** + * @file + * Contains \Drupal\aggregator\Tests\Views\IntegrationTest. + */ + +namespace Drupal\aggregator\Tests\Views; + +use Drupal\views\Tests\ViewTestData; +use Drupal\views\Tests\ViewUnitTestBase; + +/** + * Tests basic views integration of aggregator module. + */ +class IntegrationTest extends ViewUnitTestBase { + + /** + * Modules to enable. + * + * @var array + */ + public static $modules = array('aggregator', 'aggregator_test_views', 'system', 'field'); + + /** + * Views used by this test. + * + * @var array + */ + public static $testViews = array('test_aggregator_items'); + + /** + * The entity storage controller for aggregator items. + * + * @var \Drupal\aggregator\ItemStorageController + */ + protected $itemStorageController; + + /** + * The entity storage controller for aggregator feeds. + * + * @var \Drupal\aggregator\FeedStorageController + */ + protected $feedStorageController; + + public static function getInfo() { + return array( + 'name' => 'Aggregator: Integration tests', + 'description' => 'Tests basic integration of views data from the aggregator module.', + 'group' => 'Views module integration', + ); + } + + protected function setUp() { + parent::setUp(); + + $this->installSchema('aggregator', array('aggregator_item', 'aggregator_feed', 'aggregator_category_feed', 'aggregator_category', 'aggregator_category_item')); + + ViewTestData::importTestViews(get_class($this), array('aggregator_test_views')); + + $this->itemStorageController = $this->container->get('plugin.manager.entity')->getStorageController('aggregator_item'); + $this->feedStorageController = $this->container->get('plugin.manager.entity')->getStorageController('aggregator_feed'); + } + + /** + * Tests basic aggregator_item view. + */ + public function testAggregatorItemView() { + $items = array(); + $expected = array(); + for ($i = 0; $i < 10; $i++) { + $values = array(); + $values['timestamp'] = mt_rand(REQUEST_TIME - 10, REQUEST_TIME + 10); + $values['title'] = $this->randomName(); + $values['description'] = $this->randomName(); + // Add a image to ensure that the sanitizing can be tested below. + $values['author'] = $this->randomName() . '<img src="http://example.com/example.png" \>"'; + $values['link'] = 'http://drupal.org/node/' . mt_rand(1000, 10000); + + $aggregator_item = $this->itemStorageController->create($values); + $aggregator_item->save(); + $items[$aggregator_item->id()] = $aggregator_item; + + $values['iid'] = $aggregator_item->id(); + $expected[] = $values; + } + + $view = views_get_view('test_aggregator_items'); + $this->executeView($view); + + $column_map = array( + 'iid' => 'iid', + 'aggregator_item_title' => 'title', + 'aggregator_item_timestamp' => 'timestamp', + 'aggregator_item_description' => 'description', + 'aggregator_item_author' => 'author', + ); + $this->assertIdenticalResultset($view, $expected, $column_map); + + // Ensure that the rendering of the linked title works as expected. + foreach ($view->result as $row) { + $iid = $view->field['iid']->getValue($row); + $expected_link = l($items[$iid]->title->value, $items[$iid]->link->value, array('absolute' => TRUE)); + $this->assertEqual($view->field['title']->advancedRender($row), $expected_link, 'Ensure the right link is generated'); + + $expected_author = aggregator_filter_xss($items[$iid]->author->value); + $this->assertEqual($view->field['author']->advancedRender($row), $expected_author, 'Ensure the author got filtered'); + + $expected_description = aggregator_filter_xss($items[$iid]->description->value); + $this->assertEqual($view->field['description']->advancedRender($row), $expected_description, 'Ensure the author got filtered'); + } + } + +} diff --git a/core/modules/aggregator/tests/modules/aggregator_test_views/aggregator_test_views.info.yml b/core/modules/aggregator/tests/modules/aggregator_test_views/aggregator_test_views.info.yml new file mode 100644 index 000000000000..2165bd791cfd --- /dev/null +++ b/core/modules/aggregator/tests/modules/aggregator_test_views/aggregator_test_views.info.yml @@ -0,0 +1,9 @@ +name: 'Aggregator test views' +description: 'Provides default views for views aggregator tests.' +package: Testing +version: VERSION +core: 8.x +dependencies: + - aggregator + - views +hidden: true diff --git a/core/modules/aggregator/tests/modules/aggregator_test_views/aggregator_test_views.module b/core/modules/aggregator/tests/modules/aggregator_test_views/aggregator_test_views.module new file mode 100644 index 000000000000..b3d9bbc7f371 --- /dev/null +++ b/core/modules/aggregator/tests/modules/aggregator_test_views/aggregator_test_views.module @@ -0,0 +1 @@ +<?php diff --git a/core/modules/aggregator/tests/modules/aggregator_test_views/test_views/views.view.test_aggregator_items.yml b/core/modules/aggregator/tests/modules/aggregator_test_views/test_views/views.view.test_aggregator_items.yml new file mode 100755 index 000000000000..c4e51f76af93 --- /dev/null +++ b/core/modules/aggregator/tests/modules/aggregator_test_views/test_views/views.view.test_aggregator_items.yml @@ -0,0 +1,168 @@ +base_field: iid +base_table: aggregator_item +core: 8.x +description: '' +status: '1' +display: + default: + display_plugin: default + id: default + display_title: Master + position: '' + display_options: + access: + type: none + cache: + type: none + query: + type: views_query + exposed_form: + type: basic + pager: + type: full + style: + type: default + row: + type: fields + fields: + iid: + table: aggregator_item + field: iid + id: iid + plugin_id: numeric + title: + table: aggregator_item + field: title + id: title + plugin_id: aggregator_title_link + alter: + alter_text: '0' + text: '' + make_link: '0' + path: '' + absolute: '0' + external: '0' + replace_spaces: '0' + path_case: none + trim_whitespace: '0' + alt: '' + rel: '' + link_class: '' + prefix: '' + suffix: '' + target: '' + nl2br: '0' + max_length: '' + word_boundary: '1' + ellipsis: '1' + more_link: '0' + more_link_text: '' + more_link_path: '' + strip_tags: '0' + trim: '0' + preserve_tags: '' + html: '0' + timestamp: + table: aggregator_item + field: timestamp + id: timestamp + plugin_id: date + author: + table: aggregator_item + field: author + id: author + plugin_id: aggregator_xss + alter: + alter_text: '0' + text: '' + make_link: '0' + path: '' + absolute: '0' + external: '0' + replace_spaces: '0' + path_case: none + trim_whitespace: '0' + alt: '' + rel: '' + link_class: '' + prefix: '' + suffix: '' + target: '' + nl2br: '0' + max_length: '' + word_boundary: '1' + ellipsis: '1' + more_link: '0' + more_link_text: '' + more_link_path: '' + strip_tags: '0' + trim: '0' + preserve_tags: '' + html: '0' + description: + id: description + table: aggregator_item + field: description + relationship: none + group_type: group + admin_label: '' + label: Body + exclude: '0' + alter: + alter_text: '0' + text: '' + make_link: '0' + path: '' + absolute: '0' + external: '0' + replace_spaces: '0' + path_case: none + trim_whitespace: '0' + alt: '' + rel: '' + link_class: '' + prefix: '' + suffix: '' + target: '' + nl2br: '0' + max_length: '' + word_boundary: '1' + ellipsis: '1' + more_link: '0' + more_link_text: '' + more_link_path: '' + strip_tags: '0' + trim: '0' + preserve_tags: '' + html: '0' + element_type: '' + element_class: '' + element_label_type: '' + element_label_class: '' + element_label_colon: '1' + element_wrapper_type: '' + element_wrapper_class: '' + element_default_classes: '1' + empty: '' + hide_empty: '0' + empty_zero: '0' + hide_alter_empty: '1' + plugin_id: aggregator_xss + filters: { } + sorts: { } + feed_1: + display_plugin: feed + id: feed_1 + display_title: Feed + position: '' + display_options: + path: test-aggregator-items-feed + row: + type: aggregator_rss + options: + item_length: default +label: test_aggregator_items +module: views +id: test_aggregator_items +tag: '' +langcode: en -- GitLab