Skip to content

Media bundle

Introduction

The media bundle provide an api to handle and store files. With the file entity, it is easy to reference files in others entities. The bundle provide also function to change images.

Installation

bash
$ composer require enhavo/media-bundle
bash
$ yarn add @enhavo/media
yaml
# assets/admin/container.di.yaml
imports:
    - path: '@enhavo/media/services/admin/*'
yaml
# assets/theme/container.di.yaml
imports:
    - path: '@enhavo/media/services/theme/*'

File entity

The File entity or FileInterface represents a single file. As it is an entity, it is easy to attach a file to other entities. Just add a doctrine mapping.

php
namespace App\Entity;

use Doctrine\Common\Collections\ArrayCollection;
use Doctrine\ORM\Mapping as ORM;
use Enhavo\Bundle\MediaBundle\Model\FileInterface;

#[ORM\Entity]
class Book
{
    #[ORM\ManyToOne(targetEntity: FileInterface::class, cascade: ['all'])]
    public FileInterface $cover;

    #[ORM\ManyToMany(targetEntity: FileInterface::class, cascade: ['all'])]
    #[ORM\JoinTable(name: 'app_book_images')]
    public Collection $images;
    
    public function __construct() {
        $this->images = new ArrayCollection();
    }
}

Warning

Because the owning side should always be the entity that reference the file, you can't use oneToMany. For multiple files you should use ManyToMany instead!

If a file should be uploaded by a user, you can easily use the Media form type.

php
namespace App\Form\Type;

use Symfony\Component\Form\AbstractType;
use Enhavo\Bundle\MediaBundle\Form\Type\MediaType;

class BookType extends AbstractType
{
    public function buildForm(FormBuilderInterface $builder, array $options)
    {
        $builder->add('cover', MediaType::class);
        
        $builder->add('images', MediaType::class, [
            'multiple' => true,
        ])
    }
}

Or use the FileFactory to create a file and store it an entity. The Factory comes with a few functions to create a File from different sources. If you need the FileFactory in a service, you can just inject it.

php
use App\Entity\Book;
use Enhavo\Bundle\MediaBundle\Factory\FileFactory;
use Enhavo\Bundle\MediaBundle\Model\FileInterface;
use Enhavo\Bundle\ResourceBundle\Resource\ResourceManager;

/** @var FileFactory $fileFactory */
/** @var FileInterface $otherFile */
/** @var ResourceManager $resourceManager */
/** @var Book $book */

// download a file
$file = $fileFactory->createFromUri('https://domain.tld/file.png');

// create from path
$file = $fileFactory->createFromPath('/path/to/file.png');

// copy from another file
$file = $fileFactory->createFromFile($otherFile);

// attach file to entity
$book->cover = $file;

// save the book
$resourceManager->save($book);

// to store only the file you can save the file
// directly with the resource manager
$resourceManager->save($file);

Storage

As we already learn, the file entity represents a single file and therefor it can be saved to the database, but where are the content of the file will be stored? First of all, if you create a file entity, weather by the factory or form, it will never touch or move the original file. If the file entity will be saved to the database, the storage will copy the file to an internal storage and then the file entity reference to that stored file. If you retrieve a file entity from the database, it will reference to the stored file automatically. This is all done by doctrine hooks, so this happens also if you work with doctrine directly.

Format

File formats are variants of media files. In order to keep the original file untouched, we use formats to change the content of files and save them separate.

To create a format we apply filters on the original file. If we want to use a set of different filters, we chain them together.

Enhavo comes already with a set of filters, that covers most of use cases. That includes resizing, compression and preview images out of pdfs and videos. For a full list of all filters check reference

FormType

The media bundle comes with the MediaType form type, to make it easy to build a upload form.

php
use Enhavo\Bundle\MediaBundle\Form\Type\MediaType;


public function buildForm(FormBuilderInterface $builder, array $options)
{
    $builder->add('file', MediaType::class);
}

Configuration

By default, the MediaType uses the default configuration, which you can change to your own needs. You can also use multiple configurations and attach these configs to the MediaType.

yaml
# config/packages/enhavo_media.yaml
enhavo_media:
    form:
        # default config, which can be overwritten
       default:
           upload: true
           route: enhavo_media_admin_api_file_upload
           actions_file:
               download:
                   type: media_download
               format:
                   type: media_format
           actions: []
           parameters_type: Enhavo\Bundle\MediaBundle\Form\Type\FileParametersType
           parameters_options: []
       # add your own config key
       my_custom_config:
            parameters_type: App\Form\Type\FileParametersType

Use the config key my_custom_config as option of MediaType.

php
use Enhavo\Bundle\MediaBundle\Form\Type\MediaType;


public function buildForm(FormBuilderInterface $builder, array $options)
{
    $builder->add('file', MediaType::class, [
        'config' => 'my_custom_config'
    ]);
}

It is also possible to directly overwrite the config in MediaType.

php
use Enhavo\Bundle\MediaBundle\Form\Type\MediaType;


public function buildForm(FormBuilderInterface $builder, array $options)
{
    $builder->add('file', MediaType::class, [
        'config' => 'my_custom_config',
        'upload' => false, 
        'actions_file' => [], 
    ]);
}

Commands

The Media bundle provide a few command line tools.

Refreshing

If you change filter setting or the code was changed. You need to refresh your formats.

bash
app/console enhavo:media:refresh-format --format="myFormatName"

You can also use the --id option to target a file.