ImageValidator.php 7.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222
  1. <?php
  2. /**
  3. * @link https://www.yiiframework.com/
  4. * @copyright Copyright (c) 2008 Yii Software LLC
  5. * @license https://www.yiiframework.com/license/
  6. */
  7. namespace yii\validators;
  8. use Yii;
  9. use yii\helpers\Json;
  10. use yii\web\UploadedFile;
  11. /**
  12. * ImageValidator verifies if an attribute is receiving a valid image.
  13. *
  14. * @author Taras Gudz <gudz.taras@gmail.com>
  15. * @since 2.0
  16. */
  17. class ImageValidator extends FileValidator
  18. {
  19. /**
  20. * @var string the error message used when the uploaded file is not an image.
  21. * You may use the following tokens in the message:
  22. *
  23. * - {attribute}: the attribute name
  24. * - {file}: the uploaded file name
  25. */
  26. public $notImage;
  27. /**
  28. * @var int|null the minimum width in pixels.
  29. * Defaults to null, meaning no limit.
  30. * @see underWidth for the customized message used when image width is too small.
  31. */
  32. public $minWidth;
  33. /**
  34. * @var int|null the maximum width in pixels.
  35. * Defaults to null, meaning no limit.
  36. * @see overWidth for the customized message used when image width is too big.
  37. */
  38. public $maxWidth;
  39. /**
  40. * @var int|null the minimum height in pixels.
  41. * Defaults to null, meaning no limit.
  42. * @see underHeight for the customized message used when image height is too small.
  43. */
  44. public $minHeight;
  45. /**
  46. * @var int|null the maximum width in pixels.
  47. * Defaults to null, meaning no limit.
  48. * @see overHeight for the customized message used when image height is too big.
  49. */
  50. public $maxHeight;
  51. /**
  52. * @var string the error message used when the image is under [[minWidth]].
  53. * You may use the following tokens in the message:
  54. *
  55. * - {attribute}: the attribute name
  56. * - {file}: the uploaded file name
  57. * - {limit}: the value of [[minWidth]]
  58. */
  59. public $underWidth;
  60. /**
  61. * @var string the error message used when the image is over [[maxWidth]].
  62. * You may use the following tokens in the message:
  63. *
  64. * - {attribute}: the attribute name
  65. * - {file}: the uploaded file name
  66. * - {limit}: the value of [[maxWidth]]
  67. */
  68. public $overWidth;
  69. /**
  70. * @var string the error message used when the image is under [[minHeight]].
  71. * You may use the following tokens in the message:
  72. *
  73. * - {attribute}: the attribute name
  74. * - {file}: the uploaded file name
  75. * - {limit}: the value of [[minHeight]]
  76. */
  77. public $underHeight;
  78. /**
  79. * @var string the error message used when the image is over [[maxHeight]].
  80. * You may use the following tokens in the message:
  81. *
  82. * - {attribute}: the attribute name
  83. * - {file}: the uploaded file name
  84. * - {limit}: the value of [[maxHeight]]
  85. */
  86. public $overHeight;
  87. /**
  88. * {@inheritdoc}
  89. */
  90. public function init()
  91. {
  92. parent::init();
  93. if ($this->notImage === null) {
  94. $this->notImage = Yii::t('yii', 'The file "{file}" is not an image.');
  95. }
  96. if ($this->underWidth === null) {
  97. $this->underWidth = Yii::t('yii', 'The image "{file}" is too small. The width cannot be smaller than {limit, number} {limit, plural, one{pixel} other{pixels}}.');
  98. }
  99. if ($this->underHeight === null) {
  100. $this->underHeight = Yii::t('yii', 'The image "{file}" is too small. The height cannot be smaller than {limit, number} {limit, plural, one{pixel} other{pixels}}.');
  101. }
  102. if ($this->overWidth === null) {
  103. $this->overWidth = Yii::t('yii', 'The image "{file}" is too large. The width cannot be larger than {limit, number} {limit, plural, one{pixel} other{pixels}}.');
  104. }
  105. if ($this->overHeight === null) {
  106. $this->overHeight = Yii::t('yii', 'The image "{file}" is too large. The height cannot be larger than {limit, number} {limit, plural, one{pixel} other{pixels}}.');
  107. }
  108. }
  109. /**
  110. * {@inheritdoc}
  111. */
  112. protected function validateValue($value)
  113. {
  114. $result = parent::validateValue($value);
  115. return empty($result) ? $this->validateImage($value) : $result;
  116. }
  117. /**
  118. * Validates an image file.
  119. * @param UploadedFile $image uploaded file passed to check against a set of rules
  120. * @return array|null the error message and the parameters to be inserted into the error message.
  121. * Null should be returned if the data is valid.
  122. */
  123. protected function validateImage($image)
  124. {
  125. if (false === ($imageInfo = getimagesize($image->tempName))) {
  126. return [$this->notImage, ['file' => $image->name]];
  127. }
  128. list($width, $height) = $imageInfo;
  129. if ($width == 0 || $height == 0) {
  130. return [$this->notImage, ['file' => $image->name]];
  131. }
  132. if ($this->minWidth !== null && $width < $this->minWidth) {
  133. return [$this->underWidth, ['file' => $image->name, 'limit' => $this->minWidth]];
  134. }
  135. if ($this->minHeight !== null && $height < $this->minHeight) {
  136. return [$this->underHeight, ['file' => $image->name, 'limit' => $this->minHeight]];
  137. }
  138. if ($this->maxWidth !== null && $width > $this->maxWidth) {
  139. return [$this->overWidth, ['file' => $image->name, 'limit' => $this->maxWidth]];
  140. }
  141. if ($this->maxHeight !== null && $height > $this->maxHeight) {
  142. return [$this->overHeight, ['file' => $image->name, 'limit' => $this->maxHeight]];
  143. }
  144. return null;
  145. }
  146. /**
  147. * {@inheritdoc}
  148. */
  149. public function clientValidateAttribute($model, $attribute, $view)
  150. {
  151. ValidationAsset::register($view);
  152. $options = $this->getClientOptions($model, $attribute);
  153. return 'yii.validation.image(attribute, messages, ' . Json::htmlEncode($options) . ', deferred);';
  154. }
  155. /**
  156. * {@inheritdoc}
  157. */
  158. public function getClientOptions($model, $attribute)
  159. {
  160. $options = parent::getClientOptions($model, $attribute);
  161. $label = $model->getAttributeLabel($attribute);
  162. if ($this->notImage !== null) {
  163. $options['notImage'] = $this->formatMessage($this->notImage, [
  164. 'attribute' => $label,
  165. ]);
  166. }
  167. if ($this->minWidth !== null) {
  168. $options['minWidth'] = $this->minWidth;
  169. $options['underWidth'] = $this->formatMessage($this->underWidth, [
  170. 'attribute' => $label,
  171. 'limit' => $this->minWidth,
  172. ]);
  173. }
  174. if ($this->maxWidth !== null) {
  175. $options['maxWidth'] = $this->maxWidth;
  176. $options['overWidth'] = $this->formatMessage($this->overWidth, [
  177. 'attribute' => $label,
  178. 'limit' => $this->maxWidth,
  179. ]);
  180. }
  181. if ($this->minHeight !== null) {
  182. $options['minHeight'] = $this->minHeight;
  183. $options['underHeight'] = $this->formatMessage($this->underHeight, [
  184. 'attribute' => $label,
  185. 'limit' => $this->minHeight,
  186. ]);
  187. }
  188. if ($this->maxHeight !== null) {
  189. $options['maxHeight'] = $this->maxHeight;
  190. $options['overHeight'] = $this->formatMessage($this->overHeight, [
  191. 'attribute' => $label,
  192. 'limit' => $this->maxHeight,
  193. ]);
  194. }
  195. return $options;
  196. }
  197. }