lch / media-bundle
Symfony LchMediaBundle
Installs: 673
Dependents: 1
Suggesters: 0
Security: 0
Stars: 1
Watchers: 6
Forks: 2
Open Issues: 7
Type:symfony-bundle
Requires
- php: ^7.2
- ext-imagick: *
- friendsofsymfony/jsrouting-bundle: ^2.3
- knplabs/doctrine-behaviors: ^2.0
- lch/components-bundle: ^1.2
- symfony/framework-bundle: ^4.4
- twig/extensions: 1.5.*
README
This bundle brings to you a comprehensive and code-close way of handling media for Symfony 3. Features:
- Defines media types in
config.yml
- Provides events to hook on all process (CRUD)
- Provide form types to interact easily in you admin (pick an existing media/creates a new one)
- Provides Twig extension to ease media display and manipulation in Twig files (get URLs, thumbnails...)
- Provides validators to restrain media to boundaries (size, resolution for images, extensions...)
Installation and pre-requisites
Imagick is used to generate thumbnails and images sizes sets. On the GUI side, the bundle use Bootstrap 4 and jQuery. Be sure those 2 dependencies are fullfilled, especially on admin screens (media selection/creation)
For installing, use simply :
composer require lch/media-bundle
Configuration and usage
- General explanations
- Media types declaration
- Twig extensions and tools
- Events
- Form types
- Validators
- Storage strategy
- Image sizes
- Practical use cases
General explanations
Out of the box, MediaBundle defines 2 types : image and pdf. You can use those types as base for you custom ones.
Below is shown types and available fields :
Media types declaration
You need to define your media types in config.yml
. You can define as many types as you need, using the following syntax :
lch_media: types: image: name: 'your_project.image.name' # the translated name for front presentation entity: 'YourBundle\Entity\Media\Image' # the entity to be used form: 'YourBundle\Form\Media\ImageType' # the form to be used when adding media add_view: 'YourBundle/Media/Image/fragments:add.html.twig' # the add form view to be used when adding media thumbnail_view: 'YourBundle/Media/Resource/fragments:thumbnail.html.twig' # the view used for displaying thumbnail list_item_view: 'YourBundle/Media/Resource/fragments:list.item.html.twig' # the view used for displaying list item in selection lists extensions: ['jpg', 'jpeg', 'png', 'gif'] # allowed extensions max_items_per_page: 20 # not required, 20 per default thumbnail_sizes: news: width: 396 height: 227
Let's review an example for each given key :
Entity
Minimal class for above declared image could be :
<?php namespace YourBundle\Bundle\Entity\Media; use Doctrine\ORM\Mapping as ORM; use Knp\DoctrineBehaviors\Model\Blameable\Blameable; use Knp\DoctrineBehaviors\Model\Timestampable\Timestampable; use Lch\MediaBundle\Entity\Image as BaseImage; /** * Image * * @ORM\Table(name="image") * @ORM\Entity(repositoryClass="YourBundle\Repository\Media\ImageRepository") */ class Image extends BaseImage { use Blameable, Timestampable; }
It extends Lch\MediaBundle\Entity\Image
. If you want to start from scratch, you have to extends from Media
and use Storable
behavior in order to trigger all stuff link to file storage in the bundle.
Form
Minimal form class could be :
<?php namespace YourBundle\Form\Media; use Symfony\Component\Form\FormBuilderInterface; use Symfony\Component\OptionsResolver\OptionsResolver; use Lch\MediaBundle\Form\ImageType as BaseImageType; // Extends BaseImageType here for overriding constants class ImageType extends BaseImageType { /** * BaseImageType defines a NAME constant for generic image type. You override it here with your type name */ const NAME = 'your_image_type'; /** * Same for root translation path to be used in your particular type case */ const ROOT_TRANSLATION_PATH = 'your.back.media.form.image'; /** * @inheritdoc */ public function buildForm(FormBuilderInterface $builder, array $options) { // Explicit parent call is required for constant overriding parent::buildForm($builder, $options); } /** * @inheritdoc */ public function configureOptions(OptionsResolver $resolver) { $resolver->setDefaults([ 'data_class' => 'YourBundle\Entity\Media\Image' ]); } /** * @inheritdoc */ public function getParent() { return BaseImageType::class; } }
Add view
You can find below the add view for generic Image defined by the bundle :
{% form_theme mediaForm 'bootstrap_3_layout.html.twig' %} {{ form_errors(mediaForm) }} {{ form_start(mediaForm) }} {{ form_row(mediaForm.file) }} {{ form_row(mediaForm.name) }} {{ form_row(mediaForm.alt) }} <div class="text-right col-xs-12"> {{ form_row(mediaForm.submit) }} </div> {{ form_rest(mediaForm) }} {{ form_end(mediaForm) }}
If you define your own, you have to use mediaForm
as the form variable.
Thumbnail view
You can find below the thumbnail view for generic Image defined by the bundle :
<img src="{{ thumbnailEvent.thumbnailPath }}" alt="{{ thumbnailEvent.media.alt }}" />
Note : as indicated below, most of logic is event related. Thumbnail generation is one those things, so access to thumbnail data goes through event object
List item view
You can find below the list item view for generic Image defined by the bundle. You will find further explanations on twig methods below
{% set attrs = "" %} {% for key, attr in attributes %} {% set attrs = attrs ~ " " ~ key ~ '=' ~ attr %} {% endfor %} {% set size = listItemEvent.media.file.getSize()/1000 %} <div {{ attrs }} class="{% if attributes.fullSize is defined and attributes.fullSize == true %}col-xs-12{% else %}col-xs-6 col-sm-4 col-md-3{% endif %} media" data-id="{{ listItemEvent.media.id }}" data-type="{{ getClass(listItemEvent.media) }}" data-url="{{ get_url(listItemEvent.media) }}" data-name="{{ listItemEvent.media.name }}" data-width="{{ listItemEvent.media.width }}" data-height="{{ listItemEvent.media.height }}" data-size="{{ size }}" > <div class="col-xs-4"> {{ get_thumbnail(listItemEvent.media) }} </div> <div class="col-xs-8"> <p>{{ listItemEvent.media.name }}</p> <p><strong>{{ listItemEvent.media.width }}</strong>px x <strong>{{ listItemEvent.media.height }}</strong>px</p> <p><strong>{{ size }} Ko</strong></p> </div> </div>
Note : as indicated below, most of logic is event related. Thumbnail generation is one those things, so access to thumbnail data goes through event object
Extensions
Here you define extensions allowed for this media type, as an array.
Thumbnail sizes
More informations in dedicated section
Twig extension & tools
{{ get_list_item(media, attributes) }}
: display a list item{{ get_thumbnail(media, attributes) }}
: display the media thumbnail (called in getListItem){{ get_thumbnail_url(media, attributes) }}
: return the thumbnail URL only (whereget_thumbnail
will return an HTMLtag)
{{ get_url(media, attributes) }}
: returns the URL for the given media. direct media URL by default, but you can easily hook on DOWNLOAD event to return something more complicated (such as downloader for private resources)
You can find below a graphical render for several methods listed above :
Events
You can find below the complete event list thrown by the bundle (listed in Lch\Media\LchMediaEvents
) :
LchMediaEvents::DOWNLOAD
: fired byMediaController
before preparing response to deliver file, but after security checkLchMediaEvents::LIST_ITEM
: fired byMediaManager
when Twig method{{ getListItem(media, attributes) }}
is calledLchMediaEvents::PRE_DELETE
: fired byMediaController
before media deletionLchMediaEvents::PRE_PERSIST
: fired byMediaController
before media persistanceLchMediaEvents::PRE_SEARCH
: fired byMediaManager
on every list callsLchMediaEvents::POST_DELETE
: fired byMediaController
after media deletionLchMediaEvents::POST_PERSIST
: fired byMediaController
after media persistsLchMediaEvents::POST_SEARCH
: fired byMediaManager
on every list callsLchMediaEvents::REVERSE_TRANSFORM
: used byAddOrChooseMediaType
form typeLchMediaEvents::SEARCH_FORM
: fired byMediaManager
on list call to get specific input type in order to contextualize search form to media typeLchMediaEvents::PRE_STORAGE
: fired byMediaController
before media file storageLchMediaEvents::POST_STORAGE
: fired byMediaController
after media file storageLchMediaEvents::THUMBNAIL
: fired byMediaManager
when Twig method{{ getThumbnail(media, attributes) }}
is calledLchMediaEvents::TRANSFORM
: used byAddOrChooseMediaType
form typeLchMediaEvents::URL
: fired byMediaManager
when Twig method{{ getUrl(media, attributes) }}
is called
Form types
The bundle provides 2 form types for easing media selection/creation. First of all, be assured to add the correct form theme file in your admin twig files, which is LchMediaBundle:form:fields.html.twig
AddOrChooseMediaType
Here is a classical use :
$builder ->add('myImage', AddOrChooseMediaType::class, [ 'entity_reference' => Image::class, 'label' => static::ROOT_TRANSLATION_PATH . ".my_image.label", 'modal_title' => static::ROOT_TRANSLATION_PATH . '.my_image.modal.title', 'image_width' => 1499, 'image_height' => 415, 'required' => false, 'attr' => [ 'helper' => static::ROOT_TRANSLATION_PATH . '.my_image.helper' ] ]) ;
Although it's quite clear, note that entity_reference
is the media class you want to link here. Regarding validation, you can pass
image_width
andimage_height
for exact image size requiredmin_image_width
,max_image_width
,min_image_height
,max_image_height
for image size boundariesmin_media_weight
andmax_media_weight
for media size
Have a look to validators section in order to have more details.
You must then register a Twig namespace, and add the bundle field.html.twig
as a form_theme
:
twig: #... paths: #... '%kernel.project_dir%/vendor/lch/media-bundle/Resources/views': LchMediaBundle form_themes: #... - 'bootstrap_4_layout.html.twig' - '@LchMediaBundle\form\fields.html.twig'
Then, assuming you added form_theme as stated above, the twig parent form type become :
{% block css %} {{ parent() }} {# Media #} <link rel="stylesheet" href="{{ asset('bundles/lchmedia/css/media.css') }}" type="text/css" /> {% endblock css %} {% block your_content %} { form_start(form) }} {{ form_errors(form) }} {# ... #} {{ form_row(form.myImage) }} {# ... #} {% endblock your_content %} {% block scripts %} {{ parent() }} {# Routing : needed for interaction in the media popin #} <script src="{{ asset('bundles/fosjsrouting/js/router.js') }}"></script> <script src="{{ path('fos_js_routing_js', {'callback': 'fos.Router.setData'}) }}"></script> {# Medias #} <script src="{{ asset('bundles/lchmedia/js/isotope.js') }}"></script> <script src="{{ asset('bundles/lchmedia/js/jquery.media.js') }}"></script> <script src="{{ asset('bundles/lchmedia/js/media-search.js') }}"></script> {% endblock scripts %}
You have to add those 3 javascript files in order to make things work.
- The list use isotope to make elegant item presentation
- All logic (list/creation) is handled via jQuery plugin
jquery.media.js
media-search.js
contains JS logic around search (externalized to be used on a specific library page)
Result (after custom styling):
Button chooser in parent form :
After click on button, media popin appears with media chooser among available (limited to entity_reference
you provided)
You can also add a media from here (limited to entity_reference
you provided, and using form
and add_view
you provided for media type in config.yml
)
Note : as you can see in LchMediaBundle:form:fields.html.twig
, we postfixed all relevant HTML input ids with a random unique number, therefore safely allowing multiple media type usage in same form.
AddOrChooseMultipleMediasType
This type is useful to select a collection of media. It only makes the AddOrChooseMediaType
repeatable
$builder ->add('resources', AddOrChooseMultipleMediasType::class, [ 'label' => static::ROOT_TRANSLATION_PATH . '.public_documents.label', 'allow_add' => true, 'allow_delete' => true, 'prototype' => true, 'entry_type' => AddOrChooseMediaType::class, 'entry_options' => [ 'entity_reference' => Resource::class, 'label' => "your_project.test.label", 'modal_title' => static::ROOT_TRANSLATION_PATH . '.public_documents.document.modal.title', 'attr' => [ 'helper' => static::ROOT_TRANSLATION_PATH . '.public_documents.document.helper' ] ], 'attr' => array( 'class' => 'public-documents', 'helper' => static::ROOT_TRANSLATION_PATH . '.public_documents.helper', ), ]) ;
We suggest using the excellent Symfony collection plugin to handle collection easily. We still have to elaborate JS part a lot, so for now you have to repeat the random number change when dynamically adding media button.
Example below linked to above AddOrChooseMultipleMediasType
(with symfony collection)
$('.public-documents').collection({ up: '<a href="#" class="btn btn-default"><i class="glyphicon glyphicon-chevron-up"></i></a>', down: '<a href="#" class="btn btn-default"><i class="glyphicon glyphicon-chevron-down"></i></a>', add: '<a href="#" class="btn btn-default"><i class="glyphicon glyphicon-plus"></i></a>', remove: '<a href="#" class="btn btn-default"><i class="glyphicon glyphicon-minus"></i></a>', duplicate: '<a href="#" class="btn btn-default"><i class="glyphicon glyphicon-duplicate"></i></a>', // Change the random number to ensure uniqueness after_add: function(collection, element) { var $mediaSelector = element.find('.media-selector-container'); // Renew random ID on just added media selector $mediaSelector.lchMedia('renewRandomId'); return true; } }) );
Result (after custom styling):
Validators
You will find in Lch\MediaBundle\Validator
several built-in validators :
- HasAllowedFileExtension : for extension check based on configuration
- Weight : min/max, exact weight
- Image
- ImageSize : for image size check (width, height, min width, min height)
All validators work both on class and property level. So you need to define them on your media classes once for all :
<?php namespace YourBundle\Entity\Media; use Doctrine\ORM\Mapping as ORM; use Knp\DoctrineBehaviors\Model\Blameable\Blameable; use Knp\DoctrineBehaviors\Model\Timestampable\Timestampable; use Lch\MediaBundle\Behavior\MediaTaggable; use Lch\MediaBundle\Behavior\Storable; use Lch\MediaBundle\Entity\Media; use Lch\MediaBundle\Validator\Constraints as MediaAsset; /** * Resource * * @ORM\Table(name="resource") * @ORM\Entity(repositoryClass="YourBundle\Repository\Media\ResourceRepository") * @MediaAsset\HasAllowedFileExtension() * @MediaAsset\MediaWeight() */ class Resource extends Media { // Remember to use the Storable to ensure physical file is correctly stored use Storable, ; }
Storage strategy
TODO
Image sizes
Using Imagick, the bundle does 2 things :
- Generating thumbnails for all images
- Generating viewable thumbnails for PDF files (thumbnails for list item and previewer)
Using the correct key (thumbnail_sizes
) in media type declaration in config.yml
, you can produce as many thumbnails you need.
Note : so far, we use the image longer dimension and adapt the other one to keep homothetic transforms. Therefore, resulting images might not be in the exact good resolution. To be completed...
Practical use cases
Download control
- Declare media (see above)
- Register a Listener/Subscriber to LchMediaEvents::STORAGE to change storage for your media (example : to add a "/private/" subfolder)
- Add matching route, to force Symfony to handle request :
ipc_media_download: path: /uploads/resources/{id} defaults: { _controller: LchMediaBundle:Media:download, class: '%lch_media.types.resource.entity%' } methods: [ GET ]
- Add a Voter to restrain service
- Option : Register a Listener/Subscriber to LchMediaEvents::DOWNLOAD to control what to serve (add watermark...)
TODO : to be completed with detailled event usage examples