MediaHelper.php in Lightning Media 8.3
Same filename and directory in other branches
Namespace
Drupal\lightning_mediaFile
src/MediaHelper.phpView source
<?php
namespace Drupal\lightning_media;
use Drupal\Core\Entity\EntityTypeManagerInterface;
use Drupal\Core\File\FileSystemInterface;
use Drupal\file\Entity\File;
use Drupal\file\FileInterface;
use Drupal\file\Plugin\Field\FieldType\FileItem;
use Drupal\lightning_media\Exception\IndeterminateBundleException;
use Drupal\media\MediaInterface;
use Drupal\media\MediaTypeInterface;
/**
* Provides helper methods for dealing with media entities.
*/
class MediaHelper {
/**
* The entity type manager.
*
* @var \Drupal\Core\Entity\EntityTypeManagerInterface
*/
protected $entityTypeManager;
/**
* MediaHelper constructor.
*
* @param \Drupal\Core\Entity\EntityTypeManagerInterface $entity_type_manager
* The entity type manager.
*/
public function __construct(EntityTypeManagerInterface $entity_type_manager) {
$this->entityTypeManager = $entity_type_manager;
}
/**
* Returns all file extensions accepted by bundles that use file fields.
*
* @param bool $check_access
* (optional) Whether to filter the bundles by create access for the current
* user. Defaults to FALSE.
* @param string[] $bundles
* (optional) An array of bundle IDs from which to retrieve source field
* extensions. If omitted, all available bundles are allowed.
*
* @return string[]
* The file extensions accepted by all available bundles.
*/
public function getFileExtensions($check_access = FALSE, array $bundles = []) {
$extensions = '';
// Lightning Media overrides the media_bundle storage handler with a special
// one that adds an optional second parameter to loadMultiple().
$storage = $this->entityTypeManager
->getStorage('media_type');
$media_types = $storage
->loadMultiple($bundles ?: NULL, $check_access);
/** @var \Drupal\media\MediaTypeInterface $media_type */
foreach ($media_types as $media_type) {
$field = $media_type
->getSource()
->getSourceFieldDefinition($media_type);
// If the field is a FileItem or any of its descendants, we can consider
// it a file field. This will automatically include things like image
// fields, which extend file fields.
if (is_a($field
->getItemDefinition()
->getClass(), FileItem::class, TRUE)) {
$extensions .= $field
->getSetting('file_extensions') . ' ';
}
}
$extensions = preg_split('/,?\\s+/', rtrim($extensions));
return array_unique($extensions);
}
/**
* Returns the first media bundle that can accept an input value.
*
* @param mixed $value
* The input value.
* @param bool $check_access
* (optional) Whether to filter the bundles by create access for the current
* user. Defaults to TRUE.
* @param string[] $bundles
* (optional) A set of media bundle IDs which might match the input. If
* omitted, all available bundles are checked.
*
* @return \Drupal\media\MediaTypeInterface
* A media bundle that can accept the input value.
*
* @throws \Drupal\lightning_media\Exception\IndeterminateBundleException
* If the input value cannot be matched to exactly one media type.
*/
public function getBundleFromInput($value, $check_access = TRUE, array $bundles = []) {
$media_types = $this
->getBundlesFromInput($value, $check_access, $bundles);
if (count($media_types) === 1) {
return reset($media_types);
}
throw new IndeterminateBundleException($value, 0, NULL, $media_types);
}
/**
* Returns the media bundles that can accept an input value.
*
* @param mixed $value
* The input value.
* @param bool $check_access
* (optional) Whether to filter the bundles by create access for the current
* user. Defaults to TRUE.
* @param string[] $bundles
* (optional) A set of media bundle IDs which might match the input. If
* omitted, all available bundles are checked.
*
* @return \Drupal\media\MediaTypeInterface[]
* The media bundles that can accept the input value.
*/
public function getBundlesFromInput($value, $check_access = TRUE, array $bundles = []) {
// Lightning Media overrides the media_bundle storage handler with a special
// one that adds an optional second parameter to loadMultiple().
$media_types = $this->entityTypeManager
->getStorage('media_type')
->loadMultiple($bundles ?: NULL, $check_access);
ksort($media_types);
return array_filter($media_types, function (MediaTypeInterface $media_type) use ($value) {
$source = $media_type
->getSource();
return $source instanceof InputMatchInterface && $source
->appliesTo($value, $media_type);
});
}
/**
* Creates a media entity from an input value.
*
* @param mixed $value
* The input value.
* @param string[] $bundles
* (optional) A set of media bundle IDs which might match the input value.
* If omitted, all bundles to which the user has create access are checked.
*
* @return \Drupal\media\MediaInterface
* The unsaved media entity.
*/
public function createFromInput($value, array $bundles = []) {
/** @var \Drupal\media\MediaInterface $entity */
$entity = $this->entityTypeManager
->getStorage('media')
->create([
'bundle' => $this
->getBundleFromInput($value, TRUE, $bundles)
->id(),
]);
$field = static::getSourceField($entity);
if ($field) {
$field
->setValue($value);
}
return $entity;
}
/**
* Attaches a file entity to a media entity.
*
* @param \Drupal\media\MediaInterface $entity
* The media entity.
* @param \Drupal\file\FileInterface $file
* The file entity.
* @param int $replace
* (optional) What to do if the file already exists. Can be any of the
* constants accepted by file_move(). Defaults to
* \Drupal\Core\File\FileSystemInterface::EXISTS_RENAME.
*
* @return \Drupal\file\FileInterface|false
* The final file entity (unsaved), or FALSE if an error occurred.
*/
public static function useFile(MediaInterface $entity, FileInterface $file, $replace = NULL) {
// @todo Remove this and just use FileSystemInterface::EXISTS_REPLACE when support for older versions of core is dropped.
if (is_null($replace)) {
$replace = defined(FileSystemInterface::class . '::EXISTS_RENAME') ? FileSystemInterface::EXISTS_RENAME : constant('FILE_EXISTS_RENAME');
}
$field = static::getSourceField($entity);
$field
->setValue($file);
$destination = '';
$destination .= static::prepareFileDestination($entity);
if (substr($destination, -1) != '/') {
$destination .= '/';
}
$destination .= $file
->getFilename();
// If the core file_move() function has already been called, the file entity
// might have been replaced by another one that has the same ID, but a
// different URI. So reload the file entity to ensure we're using the most
// up-to-date URI.
/** @var \Drupal\file\FileInterface $file */
$file = File::load($file
->id());
if ($destination == $file
->getFileUri()) {
return $file;
}
else {
$file = file_move($file, $destination, $replace);
if ($file) {
$field
->setValue($file);
return $file;
}
else {
return FALSE;
}
}
}
/**
* Prepares the destination directory for a file attached to a media entity.
*
* @param \Drupal\media\MediaInterface $entity
* The media entity.
*
* @return string
* The destination directory URI.
*/
public static function prepareFileDestination(MediaInterface $entity) {
/** @var \Drupal\file\Plugin\Field\FieldType\FileItem $item */
$item = static::getSourceField($entity)
->first();
$destination = $item
->getUploadLocation();
// Support both Drupal 8.7's API and its antecedents. We need to call the
// deprecated symbols in an obscure way to prevent failures during
// deprecation testing.
if (version_compare(\Drupal::VERSION, '8.7.0', '>=')) {
$options = FileSystemInterface::CREATE_DIRECTORY | FileSystemInterface::MODIFY_PERMISSIONS;
\Drupal::service('file_system')
->prepareDirectory($destination, $options);
}
else {
$options = constant('FILE_CREATE_DIRECTORY') | constant('FILE_MODIFY_PERMISSIONS');
$function = 'file_prepare_directory';
$function($destination, $options);
}
return $destination;
}
/**
* Indicates if the media entity's type plugin supports dynamic previews.
*
* @param \Drupal\media\MediaInterface $entity
* The media entity.
*
* @return bool
* TRUE if dynamic previews are supported, FALSE otherwise.
*/
public static function isPreviewable(MediaInterface $entity) {
$plugin_definition = $entity
->getSource()
->getPluginDefinition();
return isset($plugin_definition['preview']);
}
/**
* Returns the media entity's source field item list.
*
* @param \Drupal\media\MediaInterface $entity
* The media entity.
*
* @return \Drupal\Core\Field\FieldItemListInterface|null
* The media entity's source field item list, or NULL if the media type
* plugin does not define a source field.
*/
public static function getSourceField(MediaInterface $entity) {
$field = $entity
->getSource()
->getSourceFieldDefinition($entity->bundle->entity);
return $field ? $entity
->get($field
->getName()) : NULL;
}
}
Classes
Name | Description |
---|---|
MediaHelper | Provides helper methods for dealing with media entities. |