class Twitter in Media entity Twitter 8
Provides media type plugin for Twitter.
Plugin annotation
@MediaType(
id = "twitter",
label = @Translation("Twitter"),
description = @Translation("Provides business logic and metadata for Twitter.")
)
Hierarchy
- class \Drupal\Component\Plugin\PluginBase implements DerivativeInspectionInterface, PluginInspectionInterface
- class \Drupal\media_entity\MediaTypeBase implements ContainerFactoryPluginInterface, MediaTypeInterface uses StringTranslationTrait
- class \Drupal\media_entity_twitter\Plugin\MediaEntity\Type\Twitter
- class \Drupal\media_entity\MediaTypeBase implements ContainerFactoryPluginInterface, MediaTypeInterface uses StringTranslationTrait
Expanded class hierarchy of Twitter
4 files declare their use of Twitter
- ThumbnailTest.php in tests/
src/ Kernel/ ThumbnailTest.php - TweetEmbedCodeConstraintValidator.php in src/
Plugin/ Validation/ Constraint/ TweetEmbedCodeConstraintValidator.php - TweetVisibleConstraintValidator.php in src/
Plugin/ Validation/ Constraint/ TweetVisibleConstraintValidator.php - TwitterEmbedFormatter.php in src/
Plugin/ Field/ FieldFormatter/ TwitterEmbedFormatter.php
File
- src/
Plugin/ MediaEntity/ Type/ Twitter.php, line 26
Namespace
Drupal\media_entity_twitter\Plugin\MediaEntity\TypeView source
class Twitter extends MediaTypeBase {
/**
* Config factory service.
*
* @var \Drupal\Core\Config\ConfigFactoryInterface
*/
protected $configFactory;
/**
* The renderer.
*
* @var \Drupal\Core\Render\RendererInterface
*/
protected $renderer;
/**
* The tweet fetcher.
*
* @var \Drupal\media_entity_twitter\TweetFetcherInterface
*/
protected $tweetFetcher;
/**
* The logger channel.
*
* @var \Drupal\Core\Logger\LoggerChannelInterface
*/
protected $logger;
/**
* {@inheritdoc}
*/
public static function create(ContainerInterface $container, array $configuration, $plugin_id, $plugin_definition) {
return new static($configuration, $plugin_id, $plugin_definition, $container
->get('entity_type.manager'), $container
->get('entity_field.manager'), $container
->get('config.factory'), $container
->get('renderer'), $container
->get('media_entity_twitter.tweet_fetcher'), $container
->get('logger.factory')
->get('media_entity_twitter'));
}
/**
* List of validation regular expressions.
*
* @var array
*/
public static $validationRegexp = array(
'@((http|https):){0,1}//(www\\.){0,1}twitter\\.com/(?<user>[a-z0-9_-]+)/(status(es){0,1})/(?<id>[\\d]+)@i' => 'id',
);
/**
* Constructs a new class instance.
*
* @param array $configuration
* A configuration array containing information about the plugin instance.
* @param string $plugin_id
* The plugin_id for the plugin instance.
* @param mixed $plugin_definition
* The plugin implementation definition.
* @param \Drupal\Core\Entity\EntityTypeManagerInterface $entity_type_manager
* Entity type manager service.
* @param \Drupal\Core\Entity\EntityFieldManagerInterface $entity_field_manager
* Entity field manager service.
* @param \Drupal\Core\Config\ConfigFactoryInterface $config_factory
* Config factory service.
* @param \Drupal\Core\Render\RendererInterface $renderer
* The renderer.
* @param \Drupal\media_entity_twitter\TweetFetcherInterface $tweet_fetcher
* The tweet fetcher.
* @param \Drupal\Core\Logger\LoggerChannelInterface $logger
* The logger channel.
*/
public function __construct(array $configuration, $plugin_id, $plugin_definition, EntityTypeManagerInterface $entity_type_manager, EntityFieldManagerInterface $entity_field_manager, ConfigFactoryInterface $config_factory, RendererInterface $renderer, TweetFetcherInterface $tweet_fetcher, LoggerChannelInterface $logger) {
parent::__construct($configuration, $plugin_id, $plugin_definition, $entity_type_manager, $entity_field_manager, $config_factory
->get('media_entity.settings'));
$this->configFactory = $config_factory;
$this->renderer = $renderer;
$this->tweetFetcher = $tweet_fetcher;
$this->logger = $logger;
}
/**
* {@inheritdoc}
*/
public function defaultConfiguration() {
return [
'use_twitter_api' => FALSE,
'generate_thumbnails' => FALSE,
];
}
/**
* {@inheritdoc}
*/
public function providedFields() {
$fields = array(
'id' => $this
->t('Tweet ID'),
'user' => $this
->t('Twitter user information'),
);
if ($this->configuration['use_twitter_api']) {
$fields += array(
'image' => $this
->t('Link to the twitter image'),
'image_local' => $this
->t('Copies tweet image to the local filesystem and returns the URI.'),
'image_local_uri' => $this
->t('Gets URI of the locally saved image.'),
'content' => $this
->t('This tweet content'),
'retweet_count' => $this
->t('Retweet count for this tweet'),
'profile_image_url_https' => $this
->t('Link to profile image'),
);
}
return $fields;
}
/**
* {@inheritdoc}
*/
public function getField(MediaInterface $media, $name) {
$matches = $this
->matchRegexp($media);
if (!$matches['id']) {
return FALSE;
}
// First we return the fields that are available from regex.
switch ($name) {
case 'id':
return $matches['id'];
case 'user':
if ($matches['user']) {
return $matches['user'];
}
return FALSE;
}
// If we have auth settings return the other fields.
if ($this->configuration['use_twitter_api'] && ($tweet = $this
->fetchTweet($matches['id']))) {
switch ($name) {
case 'image':
if (isset($tweet['extended_entities']['media'][0]['media_url'])) {
return $tweet['extended_entities']['media'][0]['media_url'];
}
return FALSE;
case 'image_local':
$local_uri = $this
->getField($media, 'image_local_uri');
if ($local_uri) {
if (file_exists($local_uri)) {
return $local_uri;
}
else {
$image_url = $this
->getField($media, 'image');
// @TODO: Use Guzzle, possibly in a service, for this.
$image_data = file_get_contents($image_url);
if ($image_data) {
return file_unmanaged_save_data($image_data, $local_uri, FILE_EXISTS_REPLACE);
}
}
}
return FALSE;
case 'image_local_uri':
$image_url = $this
->getField($media, 'image');
if ($image_url) {
return $this
->getLocalImageUri($matches['id'], $image_url);
}
return FALSE;
case 'content':
if (isset($tweet['text'])) {
return $tweet['text'];
}
return FALSE;
case 'retweet_count':
if (isset($tweet['retweet_count'])) {
return $tweet['retweet_count'];
}
return FALSE;
case 'profile_image_url_https':
if (isset($tweet['user']['profile_image_url_https'])) {
return $tweet['user']['profile_image_url_https'];
}
return FALSE;
}
}
return FALSE;
}
/**
* {@inheritdoc}
*/
public function buildConfigurationForm(array $form, FormStateInterface $form_state) {
$options = [];
$allowed_field_types = [
'string',
'string_long',
'link',
];
/** @var \Drupal\media_entity\MediaBundleInterface $bundle */
$bundle = $form_state
->getFormObject()
->getEntity();
foreach ($this->entityFieldManager
->getFieldDefinitions('media', $bundle
->id()) as $field_name => $field) {
if (in_array($field
->getType(), $allowed_field_types) && !$field
->getFieldStorageDefinition()
->isBaseField()) {
$options[$field_name] = $field
->getLabel();
}
}
$form['source_field'] = array(
'#type' => 'select',
'#title' => $this
->t('Field with source information'),
'#description' => $this
->t('Field on media entity that stores Twitter embed code or URL. You can create a bundle without selecting a value for this dropdown initially. This dropdown can be populated after adding fields to the bundle.'),
'#default_value' => empty($this->configuration['source_field']) ? NULL : $this->configuration['source_field'],
'#options' => $options,
);
$form['use_twitter_api'] = array(
'#type' => 'select',
'#title' => $this
->t('Whether to use Twitter api to fetch tweets or not.'),
'#description' => $this
->t("In order to use Twitter's api you have to create a developer account and an application. For more information consult the readme file."),
'#default_value' => empty($this->configuration['use_twitter_api']) ? 0 : $this->configuration['use_twitter_api'],
'#options' => array(
0 => $this
->t('No'),
1 => $this
->t('Yes'),
),
);
// @todo Evauate if this should be a site-wide configuration.
$form['consumer_key'] = array(
'#type' => 'textfield',
'#title' => $this
->t('Consumer key'),
'#default_value' => empty($this->configuration['consumer_key']) ? NULL : $this->configuration['consumer_key'],
'#states' => array(
'visible' => array(
':input[name="type_configuration[twitter][use_twitter_api]"]' => array(
'value' => '1',
),
),
),
);
$form['consumer_secret'] = array(
'#type' => 'textfield',
'#title' => $this
->t('Consumer secret'),
'#default_value' => empty($this->configuration['consumer_secret']) ? NULL : $this->configuration['consumer_secret'],
'#states' => array(
'visible' => array(
':input[name="type_configuration[twitter][use_twitter_api]"]' => array(
'value' => '1',
),
),
),
);
$form['oauth_access_token'] = array(
'#type' => 'textfield',
'#title' => $this
->t('Oauth access token'),
'#default_value' => empty($this->configuration['oauth_access_token']) ? NULL : $this->configuration['oauth_access_token'],
'#states' => array(
'visible' => array(
':input[name="type_configuration[twitter][use_twitter_api]"]' => array(
'value' => '1',
),
),
),
);
$form['oauth_access_token_secret'] = array(
'#type' => 'textfield',
'#title' => $this
->t('Oauth access token secret'),
'#default_value' => empty($this->configuration['oauth_access_token_secret']) ? NULL : $this->configuration['oauth_access_token_secret'],
'#states' => array(
'visible' => array(
':input[name="type_configuration[twitter][use_twitter_api]"]' => array(
'value' => '1',
),
),
),
);
$form['generate_thumbnails'] = [
'#type' => 'checkbox',
'#title' => $this
->t('Generate thumbnails'),
'#default_value' => $this->configuration['generate_thumbnails'],
'#states' => [
'visible' => [
':input[name="type_configuration[twitter][use_twitter_api]"]' => [
'checked' => TRUE,
],
],
],
'#description' => $this
->t('If checked, Drupal will automatically generate thumbnails for tweets that do not reference any external media. In certain circumstances, <strong>this may violate <a href="@policy">Twitter\'s fair use policy</a></strong>. Please <strong>read it and be careful</strong> if you choose to enable this.', [
'@policy' => 'https://dev.twitter.com/overview/terms/agreement-and-policy',
]),
];
return $form;
}
/**
* {@inheritdoc}
*/
public function attachConstraints(MediaInterface $media) {
parent::attachConstraints($media);
if (isset($this->configuration['source_field'])) {
$source_field_name = $this->configuration['source_field'];
if ($media
->hasField($source_field_name)) {
foreach ($media
->get($source_field_name) as &$embed_code) {
/** @var \Drupal\Core\TypedData\DataDefinitionInterface $typed_data */
$typed_data = $embed_code
->getDataDefinition();
$typed_data
->addConstraint('TweetEmbedCode');
$typed_data
->addConstraint('TweetVisible');
}
}
}
}
/**
* Computes the destination URI for a tweet image.
*
* @param mixed $id
* The tweet ID.
* @param string|null $media_url
* The URL of the media (i.e., photo, video, etc.) associated with the
* tweet.
*
* @return string
* The desired local URI.
*/
protected function getLocalImageUri($id, $media_url = NULL) {
$directory = $this->configFactory
->get('media_entity_twitter.settings')
->get('local_images');
// Ensure that the destination directory is writable. If not, log a warning
// and return the default thumbnail.
$ready = file_prepare_directory($directory, FILE_CREATE_DIRECTORY | FILE_MODIFY_PERMISSIONS);
if (!$ready) {
$this->logger
->warning('Could not prepare thumbnail destination directory @dir', [
'@dir' => $directory,
]);
return $this
->getDefaultThumbnail();
}
$local_uri = $directory . '/' . $id . '.';
if ($media_url) {
$local_uri .= pathinfo($media_url, PATHINFO_EXTENSION);
}
else {
// If there is no media associated with the tweet, we will generate an
// SVG thumbnail.
$local_uri .= 'svg';
}
return $local_uri;
}
/**
* {@inheritdoc}
*/
public function getDefaultThumbnail() {
return $this->config
->get('icon_base') . '/twitter.png';
}
/**
* {@inheritdoc}
*/
public function thumbnail(MediaInterface $media) {
// If there's already a local image, use it.
if ($local_image = $this
->getField($media, 'image_local')) {
return $local_image;
}
// If thumbnail generation is disabled, use the default thumbnail.
if (empty($this->configuration['generate_thumbnails'])) {
return $this
->getDefaultThumbnail();
}
// We might need to generate a thumbnail...
$id = $this
->getField($media, 'id');
$thumbnail_uri = $this
->getLocalImageUri($id);
// ...unless we already have, in which case, use it.
if (file_exists($thumbnail_uri)) {
return $thumbnail_uri;
}
// Render the thumbnail SVG using the theme system.
$thumbnail = [
'#theme' => 'media_entity_twitter_tweet_thumbnail',
'#content' => $this
->getField($media, 'content'),
'#author' => $this
->getField($media, 'user'),
'#avatar' => $this
->getField($media, 'profile_image_url_https'),
];
$svg = $this->renderer
->renderRoot($thumbnail);
return file_unmanaged_save_data($svg, $thumbnail_uri, FILE_EXISTS_ERROR) ?: $this
->getDefaultThumbnail();
}
/**
* Runs preg_match on embed code/URL.
*
* @param MediaInterface $media
* Media object.
*
* @return array|bool
* Array of preg matches or FALSE if no match.
*
* @see preg_match()
*/
protected function matchRegexp(MediaInterface $media) {
$matches = array();
if (isset($this->configuration['source_field'])) {
$source_field = $this->configuration['source_field'];
if ($media
->hasField($source_field)) {
$property_name = $media->{$source_field}
->first()
->mainPropertyName();
foreach (static::$validationRegexp as $pattern => $key) {
if (preg_match($pattern, $media->{$source_field}->{$property_name}, $matches)) {
return $matches;
}
}
}
}
return FALSE;
}
/**
* Get a single tweet.
*
* @param int $id
* The tweet id.
*/
protected function fetchTweet($id) {
$this->tweetFetcher
->setCredentials($this->configuration['consumer_key'], $this->configuration['consumer_secret'], $this->configuration['oauth_access_token'], $this->configuration['oauth_access_token_secret']);
try {
return $this->tweetFetcher
->fetchTweet($id);
} catch (\Exception $e) {
throw new MediaTypeException(NULL, $e
->getMessage());
}
}
/**
* {@inheritdoc}
*/
public function getDefaultName(MediaInterface $media) {
// The default name will be the twitter username of the author + the
// tweet ID.
$user = $this
->getField($media, 'user');
$id = $this
->getField($media, 'id');
if (!empty($user) && !empty($id)) {
return $user . ' - ' . $id;
}
return parent::getDefaultName($media);
}
}
Members
Name | Modifiers | Type | Description | Overrides |
---|---|---|---|---|
MediaTypeBase:: |
protected | property | Media entity image config object. | |
MediaTypeBase:: |
protected | property | The entity field manager service. | |
MediaTypeBase:: |
protected | property | The entity type manager service. | |
MediaTypeBase:: |
protected | property | Plugin label. | |
MediaTypeBase:: |
public | function |
Calculates dependencies for the configured plugin. Overrides DependentPluginInterface:: |
|
MediaTypeBase:: |
public | function |
Gets this plugin's configuration. Overrides ConfigurablePluginInterface:: |
|
MediaTypeBase:: |
public | function |
Returns the display label. Overrides MediaTypeInterface:: |
|
MediaTypeBase:: |
public | function |
Sets the configuration for this plugin instance. Overrides ConfigurablePluginInterface:: |
|
MediaTypeBase:: |
public | function |
Form submission handler. Overrides PluginFormInterface:: |
|
MediaTypeBase:: |
public | function |
Form validation handler. Overrides PluginFormInterface:: |
|
PluginBase:: |
protected | property | Configuration information passed into the plugin. | 1 |
PluginBase:: |
protected | property | The plugin implementation definition. | 1 |
PluginBase:: |
protected | property | The plugin_id. | |
PluginBase:: |
constant | A string which is used to separate base plugin IDs from the derivative ID. | ||
PluginBase:: |
public | function |
Gets the base_plugin_id of the plugin instance. Overrides DerivativeInspectionInterface:: |
|
PluginBase:: |
public | function |
Gets the derivative_id of the plugin instance. Overrides DerivativeInspectionInterface:: |
|
PluginBase:: |
public | function |
Gets the definition of the plugin implementation. Overrides PluginInspectionInterface:: |
3 |
PluginBase:: |
public | function |
Gets the plugin_id of the plugin instance. Overrides PluginInspectionInterface:: |
|
PluginBase:: |
public | function | Determines if the plugin is configurable. | |
StringTranslationTrait:: |
protected | property | The string translation service. | 1 |
StringTranslationTrait:: |
protected | function | Formats a string containing a count of items. | |
StringTranslationTrait:: |
protected | function | Returns the number of plurals supported by a given language. | |
StringTranslationTrait:: |
protected | function | Gets the string translation service. | |
StringTranslationTrait:: |
public | function | Sets the string translation service to use. | 2 |
StringTranslationTrait:: |
protected | function | Translates a string to the current language or to a given language. | |
Twitter:: |
protected | property | Config factory service. | |
Twitter:: |
protected | property | The logger channel. | |
Twitter:: |
protected | property | The renderer. | |
Twitter:: |
protected | property | The tweet fetcher. | |
Twitter:: |
public static | property | List of validation regular expressions. | |
Twitter:: |
public | function |
Attaches type-specific constraints to media. Overrides MediaTypeBase:: |
|
Twitter:: |
public | function |
Form constructor. Overrides MediaTypeBase:: |
|
Twitter:: |
public static | function |
Creates an instance of the plugin. Overrides MediaTypeBase:: |
|
Twitter:: |
public | function |
Gets default configuration for this plugin. Overrides MediaTypeBase:: |
|
Twitter:: |
protected | function | Get a single tweet. | |
Twitter:: |
public | function |
Provide a default name for the media. Overrides MediaTypeBase:: |
|
Twitter:: |
public | function |
Gets the default thumbnail image. Overrides MediaTypeBase:: |
|
Twitter:: |
public | function |
Gets a media-related field/value. Overrides MediaTypeInterface:: |
|
Twitter:: |
protected | function | Computes the destination URI for a tweet image. | |
Twitter:: |
protected | function | Runs preg_match on embed code/URL. | |
Twitter:: |
public | function |
Gets list of fields provided by this plugin. Overrides MediaTypeInterface:: |
|
Twitter:: |
public | function |
Gets thumbnail image. Overrides MediaTypeInterface:: |
|
Twitter:: |
public | function |
Constructs a new class instance. Overrides MediaTypeBase:: |