ORIENT/modules/backend/formwidgets/FileUpload.php

439 lines
11 KiB
PHP
Raw Normal View History

2014-05-14 13:24:20 +00:00
<?php namespace Backend\FormWidgets;
use Str;
use Lang;
2014-05-14 13:24:20 +00:00
use Input;
use Response;
2014-05-14 13:24:20 +00:00
use Validator;
use System\Models\File;
use ApplicationException;
2014-12-05 07:35:42 +00:00
use Backend\Classes\FormField;
2014-05-14 13:24:20 +00:00
use Backend\Classes\FormWidgetBase;
use Backend\Controllers\Files as FilesController;
2015-01-28 07:03:35 +00:00
use ValidationException;
use Exception;
2014-05-14 13:24:20 +00:00
/**
* File upload field
* Renders a form file uploader field.
*
* Supported options:
* - mode: image-single, image-multi, file-single, file-multi
* - upload-label: Add file
* - empty-label: No file uploaded
*
* @package october\backend
* @author Alexey Bobkov, Samuel Georges
*/
class FileUpload extends FormWidgetBase
{
//
// Configurable properties
//
2014-05-14 13:24:20 +00:00
2015-06-13 00:41:00 +00:00
/**
* @var string Prompt to display if no record is selected.
*/
public $prompt = 'Click the %s or drag a file here to upload';
/**
* @var int Preview image width
*/
public $imageWidth = null;
/**
* @var int Preview image height
*/
public $imageHeight = null;
/**
* @var string Text to display when no file is associated
*/
public $previewNoFilesMessage = 'backend::lang.form.preview_no_files_message';
2014-05-14 13:24:20 +00:00
/**
* @var mixed Collection of acceptable file types.
*/
public $fileTypes = false;
/**
* @var mixed Collection of acceptable mime types.
*/
public $mimeTypes = false;
/**
* @var array Options used for generating thumbnails.
*/
public $thumbOptions = [
'mode' => 'crop',
'extension' => 'auto'
];
/**
* @var boolean Allow the user to set a caption.
*/
public $useCaption = true;
//
// Object properties
//
/**
* {@inheritDoc}
*/
protected $defaultAlias = 'fileupload';
2014-05-14 13:24:20 +00:00
/**
* {@inheritDoc}
*/
public function init()
{
$this->fillFromConfig([
2015-06-13 00:41:00 +00:00
'prompt',
'imageWidth',
'imageHeight',
2014-10-10 21:50:05 +00:00
'previewNoFilesMessage',
'fileTypes',
'mimeTypes',
'thumbOptions',
'useCaption'
]);
2014-05-14 13:24:20 +00:00
$this->checkUploadPostback();
}
/**
* {@inheritDoc}
*/
public function render()
{
$this->prepareVars();
return $this->makePartial('fileupload');
2014-05-14 13:24:20 +00:00
}
/**
* Prepares the view data
*/
protected function prepareVars()
2014-05-14 13:24:20 +00:00
{
$this->vars['fileList'] = $fileList = $this->getFileList();
$this->vars['singleFile'] = $fileList->first();
2014-05-14 13:24:20 +00:00
$this->vars['displayMode'] = $this->getDisplayMode();
$this->vars['emptyIcon'] = $this->getConfig('emptyIcon', 'icon-plus');
$this->vars['imageHeight'] = $this->imageHeight;
$this->vars['imageWidth'] = $this->imageWidth;
$this->vars['acceptedFileTypes'] = $this->getAcceptedFileTypes(true);
$this->vars['cssDimensions'] = $this->getCssDimensions();
$this->vars['useCaption'] = $this->useCaption;
2015-06-13 00:41:00 +00:00
$this->vars['prompt'] = str_replace('%s', '<i class="icon-upload"></i>', $this->prompt);
2014-05-14 13:24:20 +00:00
}
protected function getFileList()
{
$list = $this
->getRelationObject()
->withDeferred($this->sessionKey)
->orderBy('sort_order')
->get()
;
2014-05-14 13:24:20 +00:00
2014-08-01 08:14:39 +00:00
/*
* Decorate each file with thumb and custom download path
2014-08-01 08:14:39 +00:00
*/
$list->each(function($file){
$this->decorateFileAttributes($file);
});
2014-08-01 08:14:39 +00:00
2014-05-14 13:24:20 +00:00
return $list;
}
/**
* Returns the display mode for the file upload. Eg: file-multi, image-single, etc.
* @return string
*/
2014-05-14 13:24:20 +00:00
protected function getDisplayMode()
{
$mode = $this->getConfig('mode', 'image');
2014-10-10 21:50:05 +00:00
if (str_contains($mode, '-')) {
2014-05-14 13:24:20 +00:00
return $mode;
2014-10-10 21:50:05 +00:00
}
2014-05-14 13:24:20 +00:00
$relationType = $this->getRelationType();
2014-05-14 13:24:20 +00:00
$mode .= ($relationType == 'attachMany' || $relationType == 'morphMany') ? '-multi' : '-single';
return $mode;
}
/**
* Returns the CSS dimensions for the uploaded image,
* uses auto where no dimension is provided.
* @return string
*/
protected function getCssDimensions()
{
$cssDimensions = '';
$cssDimensions .= ($this->imageWidth)
? 'width: '.$this->imageWidth.'px;'
: 'width: auto;';
$cssDimensions .= ($this->imageHeight)
? 'height: '.$this->imageHeight.'px;'
: 'height: auto;';
return $cssDimensions;
}
/**
* Returns the specified accepted file types, or the default
* based on the mode. Image mode will return:
* - jpg,jpeg,bmp,png,gif,svg
* @return string
*/
public function getAcceptedFileTypes($includeDot = false)
{
$types = $this->fileTypes;
if ($types === false) {
$isImage = starts_with($this->getDisplayMode(), 'image');
$types = implode(',', File::getDefaultFileTypes($isImage));
2015-02-14 10:38:49 +00:00
}
if (!$types || $types == '*') {
2015-02-14 10:38:49 +00:00
return null;
}
if (!is_array($types)) {
$types = explode(',', $types);
}
$types = array_map(function($value) use ($includeDot) {
$value = trim($value);
if (substr($value, 0, 1) == '.') {
$value = substr($value, 1);
}
if ($includeDot) {
$value = '.'.$value;
}
return $value;
}, $types);
return implode(',', $types);
}
/**
* Returns the value as a relation object from the model,
* supports nesting via HTML array.
* @return Relation
*/
protected function getRelationObject()
{
list($model, $attribute) = $this->resolveModelAttribute($this->valueFrom);
if (!$model->hasRelation($attribute)) {
throw new ApplicationException(Lang::get('backend::lang.model.missing_relation', [
'class' => get_class($model),
'relation' => $attribute
]));
}
return $model->{$attribute}();
}
/**
* Returns the value as a relation type from the model,
* supports nesting via HTML array.
* @return Relation
*/
protected function getRelationType()
{
list($model, $attribute) = $this->resolveModelAttribute($this->valueFrom);
return $model->getRelationType($attribute);
}
2014-05-14 13:24:20 +00:00
/**
* Removes a file attachment.
*/
public function onRemoveAttachment()
{
if (($file_id = post('file_id')) && ($file = File::find($file_id))) {
2014-09-06 11:46:19 +00:00
$this->getRelationObject()->remove($file, $this->sessionKey);
2014-05-14 13:24:20 +00:00
}
}
/**
* Sorts file attachments.
*/
public function onSortAttachments()
{
if ($sortData = post('sortOrder')) {
$ids = array_keys($sortData);
$orders = array_values($sortData);
$file = new File;
$file->setSortableOrder($ids, $orders);
}
}
/**
* Loads the configuration form for an attachment, allowing title and description to be set.
*/
public function onLoadAttachmentConfig()
{
if (($file_id = post('file_id')) && ($file = File::find($file_id))) {
$file = $this->decorateFileAttributes($file);
2014-05-14 13:24:20 +00:00
$this->vars['file'] = $file;
$this->vars['displayMode'] = $this->getDisplayMode();
$this->vars['cssDimensions'] = $this->getCssDimensions();
2014-05-14 13:24:20 +00:00
return $this->makePartial('config_form');
}
throw new ApplicationException('Unable to find file, it may no longer exist');
2014-05-14 13:24:20 +00:00
}
/**
* Commit the changes of the attachment configuration form.
*/
public function onSaveAttachmentConfig()
{
try {
if (($file_id = post('file_id')) && ($file = File::find($file_id))) {
$file->title = post('title');
$file->description = post('description');
$file->save();
return ['displayName' => $file->title ?: $file->file_name];
2014-05-14 13:24:20 +00:00
}
throw new ApplicationException('Unable to find file, it may no longer exist');
}
catch (Exception $ex) {
2014-05-14 13:24:20 +00:00
return json_encode(['error' => $ex->getMessage()]);
}
}
/**
* {@inheritDoc}
*/
public function loadAssets()
{
2014-05-24 06:57:38 +00:00
$this->addCss('css/fileupload.css', 'core');
$this->addJs('js/fileupload.js', 'core');
2014-05-14 13:24:20 +00:00
}
/**
* {@inheritDoc}
*/
2015-01-04 22:43:39 +00:00
public function getSaveValue($value)
{
2014-12-05 07:35:42 +00:00
return FormField::NO_SAVE_DATA;
}
2014-05-14 13:24:20 +00:00
/**
* Checks the current request to see if it is a postback containing a file upload
* for this particular widget.
*/
protected function checkUploadPostback()
{
2014-10-10 21:50:05 +00:00
if (!($uniqueId = post('X_OCTOBER_FILEUPLOAD')) || $uniqueId != $this->getId()) {
2014-05-14 13:24:20 +00:00
return;
2014-10-10 21:50:05 +00:00
}
2014-05-14 13:24:20 +00:00
try {
2015-04-13 11:51:22 +00:00
if (!Input::hasFile('file_data')) {
2015-04-12 09:58:28 +00:00
throw new ApplicationException('File missing from request');
}
2014-05-14 13:24:20 +00:00
2015-04-13 11:51:22 +00:00
$uploadedFile = Input::file('file_data');
2014-05-14 13:24:20 +00:00
$validationRules = ['max:'.File::getMaxFilesize()];
if ($fileTypes = $this->getAcceptedFileTypes()) {
$validationRules[] = 'extensions:'.$fileTypes;
}
if ($this->mimeTypes) {
$validationRules[] = 'mimes:'.$this->mimeTypes;
2014-10-10 21:50:05 +00:00
}
2014-05-14 13:24:20 +00:00
$validation = Validator::make(
['file_data' => $uploadedFile],
['file_data' => $validationRules]
);
2014-10-10 21:50:05 +00:00
if ($validation->fails()) {
2014-05-14 13:24:20 +00:00
throw new ValidationException($validation);
2014-10-10 21:50:05 +00:00
}
2014-05-14 13:24:20 +00:00
2014-10-10 21:50:05 +00:00
if (!$uploadedFile->isValid()) {
throw new ApplicationException('File is not valid');
2014-10-10 21:50:05 +00:00
}
2014-05-14 13:24:20 +00:00
2014-09-06 11:46:19 +00:00
$fileRelation = $this->getRelationObject();
2014-05-14 13:24:20 +00:00
$file = new File();
$file->data = $uploadedFile;
$file->is_public = $fileRelation->isPublic();
2014-05-14 13:24:20 +00:00
$file->save();
$fileRelation->add($file, $this->sessionKey);
$file = $this->decorateFileAttributes($file);
$result = [
'id' => $file->id,
'thumb' => $file->thumbUrl,
'path' => $file->pathUrl
];
Response::json($result, 200)->send();
2014-05-14 13:24:20 +00:00
}
catch (Exception $ex) {
Response::json($ex->getMessage(), 400)->send();
2014-05-14 13:24:20 +00:00
}
exit;
2014-05-14 13:24:20 +00:00
}
/**
* Adds the bespoke attributes used internally by this widget.
* - thumbUrl
* - pathUrl
* @return System\Models\File
*/
protected function decorateFileAttributes($file)
{
/*
* File is protected, create a secure public path
*/
if (!$file->isPublic()) {
$path = $thumb = FilesController::getDownloadUrl($file);
if ($this->imageWidth || $this->imageHeight) {
$thumb = FilesController::getThumbUrl($file, $this->imageWidth, $this->imageHeight, $this->thumbOptions);
}
}
/*
* Otherwise use public paths
*/
else {
$path = $thumb = $file->getPath();
if ($this->imageWidth || $this->imageHeight) {
$thumb = $file->getThumb($this->imageWidth, $this->imageHeight, $this->thumbOptions);
}
}
$file->pathUrl = $path;
$file->thumbUrl = $thumb;
return $file;
}
2014-10-10 21:50:05 +00:00
}