floor12 / yii2-module-files

Yii2 modul to add files to your models.
14 stars 19 forks source link

yii2-module-files

Scrutinizer Code Quality Latest Stable Version Latest Unstable Version Total Downloads License

Этот файл доступен на русском языке.

About the module

FileInputWidget

This module was designed to solve the problem of creating file fields in ActiveRecord models of the Yii2 framework. The main components of the module are:

Key features

i18n

At this stage, the module supports the following languages:

Principle of operation

All files data is stored in the file table. The file model relay to the model by three fields:

When file added to the form, it uploads to server in background where all processing takes place. As a result of this processing, it is written to disk and a new entry is created for it in the file table, with the fieldsclass and field filled with data from the model, andobject_id is empty and will assign only after saving the ActiveRecord model. When a file is deleted from the widget, it is not deleted from the disk and the file table, just obejct_id equals to 0. You can use the console commandfiles / console / clean to periodically clean up this kind of orphan files.

Install and setup

To add this module to your app, just run:

 $ composer require floor12/yii2-module-files

or add this to the require section of your composer.json.

 "floor12/yii2-module-files": "dev-master"

Then execute a migration to create file table.

 $ ./yii migrate --migrationPath=@vendor/floor12/yii2-module-files/src/migrations/

After that, include module data in modules section of application config:

 'modules' => [
             'files' => [
                 'class' => 'floor12\files\Module',
                 'storage' => '@app/storage',
                 'cache' => '@app/storage_cache',
                 'token_salt' => 'some_random_salt',
             ],
         ],
     ...

Parameters to set:

Usage

Work with ActiveRecord Model

To add one or more files fields to the ActiveRecord model, you need to connect floor12\files\components\FileBehaviour to it and pass list the field names that will store the files in the model. For example, for the User model, 2 file fields are defined here : avatar anddocuments:

  public function behaviors()
  {
      return [
          'files' => [
              'class' => 'floor12\files\components\FileBehaviour',
              'attributes' => [
                  'avatar',
                  'documents'
              ],
          ],
          ...

To have nice attribute labels in forms, add some labels to attributeLabels():

  public function attributeLabels()
     {
         return [
             ...
             'avatar' => 'Аватар',
             'documents' => 'Документы',
             ...
         ];
     }

Setup validation rules in the rules() method of ActiveRecord model:

 public function rules()
 {
     return [
         // Avatar is required attribute 
         ['avatar', 'required'],

         // Avatar allow to uploade 1 file with this extensions: jpg, png, jpeg, gif
         ['avatar', 'file', 'extensions' => ['jpg', 'png', 'jpeg', 'gif'], 'maxFiles' => 1], 

         // Documens allows to upload a few files with this extensions: docx, xlsx
         ['documents', 'file', 'extensions' => ['docx', 'xlsx'], 'maxFiles' => 10],
     ...    

Work with files

If maxFiles in FileValidator equals to 1, this attribute will store an floor12\files\models\File object. Example:

 // The href field contains web path to file source
 echo Html::img($model->avatar->href)     

 // __toString() method of File object will return href as well
 echo Html::img($model->avatar)          

If the file is image, getPreviewWebPath method returns a web path to image thumbnail. By default thumbnail created with the jpeg or png format, it depends to source file. But also WEBP option is available.

File::getPreviewWebPath(int $width = 0, int $height = 0 ,bool $webp = false)

Usage example:

 // User avatar thumbnail with 200px width 
 echo Html::img($model->avatar->getPreviewWebPath(200));     

 // User avatar thumbnail with 200px width  and WEBP format
 echo Html::img($model->avatar->getPreviewWebPath(200, 0, true));     

When maxFiles equals to 1, multiple upload is available. In this case, model field will contains an array if floor12\files\models \File objects:

 foreach ($model->docs as $doc}
     Html::a($doc->title, $doc->href);

Here is another example, the advanced usage of thumbnails. In this case, we use modern picture andsource tags, as well as media queries. As a result, we have 8 different thumbnails: 4 has webp format for those browsers that support this it, and 4 has jpeg format. Devices with retina displays will get an images with double resolution, regular screens have regular sized pictures. This example also uses different images widths at different screen widths (just as example of mobile/desktop image switching):

 <picture>
     <source type="image/webp" media='(min-width: 500px)' srcset="
                              <?= $model->poster->getPreviewWebPath(150, true) ?> 1x,
                              <?= $model->poster->getPreviewWebPath(300, true) ?> 2x">
     <source type="image/webp" media='(max-width: 500px)' srcset="
                              <?= $model->poster->getPreviewWebPath(350, true) ?> 1x,
                              <?= $model->poster->getPreviewWebPath(700, true) ?> 2x">
     <source type="image/jpeg" media='(min-width: 500px)' srcset="
                              <?= $model->poster->getPreviewWebPath(150, false) ?> 1x,
                              <?= $model->poster->getPreviewWebPath(300, false) ?> 2x">
     <img src="https://github.com/floor12/yii2-module-files/raw/master/<?= $model->poster->getPreviewWebPath(150) ?>" 
          srcset=" 
                <?= $model->poster->getPreviewWebPath(350) ?> 1x, 
                <?= $model->poster->getPreviewWebPath(700) ?> 2x"
          alt="<?= $model->title ?>">
 </picture>

Picture tag widget

If object if tyle File is image ($file->isImage() === true), it can be used with PictureWidget. This widget helps generate html tag with srcset with 2x and webp versions. For example this code:

echo PictureWidget::widget([
    'model' => $file,
    'alt' => 'Some alternative text',
    'width' => 100
]);

will make this html:


<picture>
    <source type="image/webp"
            srcset="/imageurl/image1xWebp 1x,/imageurl/image2xWebp 2x">
    <img src="https://github.com/floor12/yii2-module-files/raw/master/imageurl/image 1x"
         srcset="/imageurl/image 1x,/imageurl/image 2x"
         alt="<?= $model->title ?>">
</picture>

Additional parameters allowed to pass media-queries to widget.

Listing the files

There is a widget for listing all files. It supports Lightbox2 gallery to display images and MS Office files preview. Its also supports downloading of the all the files attached to the field in a ZIP-archive.

 echo \floor12\files\components\FileListWidget::widget([
     'files' => $model->docs, 
     'downloadAll' => true, 
     'zipTitle' => "Documents of {$user->fullname}" 
 ]) 

An array of File objects must be passed to the widget files field. Also additional parameters available:

FileInputWidget

InputWidget for ActiveFrom

To display files block in your forms use the floor12\files\components\FileInputWidget:

<?= $form->field($model, 'avatar')->widget(floor12\files\components\FileInputWidget::class) ?>

Moreover, if maxFiles parameter inFileValidator equals to 1 or more, FileInputWidget will take the necessary form to load one file or several at once. If necessary, you can pass uploadButtonText anduploadButtonClass parameters to the widget.

Contributing

I will be glad of any help in the development, support and bug reporting of this module.