DynamicModel.php 8.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282
  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\base;
  8. use yii\validators\Validator;
  9. /**
  10. * DynamicModel is a model class that supports defining attributes at run-time (the so-called
  11. * "dynamic attributes") using its constructor or [[defineAttribute()]]. DynamicModel can be used
  12. * to support ad hoc data validation.
  13. *
  14. * The typical usage of DynamicModel is as follows,
  15. *
  16. * ```php
  17. * public function actionSearch($name, $email)
  18. * {
  19. * $model = DynamicModel::validateData(compact('name', 'email'), [
  20. * [['name', 'email'], 'string', 'max' => 128],
  21. * ['email', 'email'],
  22. * ]);
  23. * if ($model->hasErrors()) {
  24. * // validation fails
  25. * } else {
  26. * // validation succeeds
  27. * }
  28. * }
  29. * ```
  30. *
  31. * The above example shows how to validate `$name` and `$email` with the help of DynamicModel.
  32. * The [[validateData()]] method creates an instance of DynamicModel, defines the attributes
  33. * using the given data (`name` and `email` in this example), and then calls [[Model::validate()]].
  34. *
  35. * You can check the validation result using [[hasErrors()]], like you do with a normal model.
  36. * You may also access the dynamic attributes defined through the model instance, e.g.,
  37. * `$model->name` and `$model->email`.
  38. *
  39. * Alternatively, you may use the following more "classic" syntax to perform ad-hoc data validation:
  40. *
  41. * ```php
  42. * $model = new DynamicModel(compact('name', 'email'));
  43. * $model->addRule(['name', 'email'], 'string', ['max' => 128])
  44. * ->addRule('email', 'email')
  45. * ->validate();
  46. * ```
  47. *
  48. * @author Qiang Xue <qiang.xue@gmail.com>
  49. * @since 2.0
  50. */
  51. class DynamicModel extends Model
  52. {
  53. /**
  54. * @var mixed[] dynamic attribute values (name => value).
  55. */
  56. private $_attributes = [];
  57. /**
  58. * @var string[] dynamic attribute labels (name => label).
  59. * Used as form field labels and in validation error messages.
  60. * @since 2.0.35
  61. */
  62. private $_attributeLabels = [];
  63. /**
  64. * Constructor.
  65. * @param array $attributes the attributes (name-value pairs, or names) being defined.
  66. * @param array $config the configuration array to be applied to this object.
  67. */
  68. public function __construct(array $attributes = [], $config = [])
  69. {
  70. foreach ($attributes as $name => $value) {
  71. if (is_int($name)) {
  72. $this->_attributes[$value] = null;
  73. } else {
  74. $this->_attributes[$name] = $value;
  75. }
  76. }
  77. parent::__construct($config);
  78. }
  79. /**
  80. * {@inheritdoc}
  81. */
  82. public function __get($name)
  83. {
  84. if ($this->hasAttribute($name)) {
  85. return $this->_attributes[$name];
  86. }
  87. return parent::__get($name);
  88. }
  89. /**
  90. * {@inheritdoc}
  91. */
  92. public function __set($name, $value)
  93. {
  94. if ($this->hasAttribute($name)) {
  95. $this->_attributes[$name] = $value;
  96. } else {
  97. parent::__set($name, $value);
  98. }
  99. }
  100. /**
  101. * {@inheritdoc}
  102. */
  103. public function __isset($name)
  104. {
  105. if ($this->hasAttribute($name)) {
  106. return isset($this->_attributes[$name]);
  107. }
  108. return parent::__isset($name);
  109. }
  110. /**
  111. * {@inheritdoc}
  112. */
  113. public function __unset($name)
  114. {
  115. if ($this->hasAttribute($name)) {
  116. unset($this->_attributes[$name]);
  117. } else {
  118. parent::__unset($name);
  119. }
  120. }
  121. /**
  122. * {@inheritdoc}
  123. */
  124. public function canGetProperty($name, $checkVars = true, $checkBehaviors = true)
  125. {
  126. return parent::canGetProperty($name, $checkVars, $checkBehaviors) || $this->hasAttribute($name);
  127. }
  128. /**
  129. * {@inheritdoc}
  130. */
  131. public function canSetProperty($name, $checkVars = true, $checkBehaviors = true)
  132. {
  133. return parent::canSetProperty($name, $checkVars, $checkBehaviors) || $this->hasAttribute($name);
  134. }
  135. /**
  136. * Returns a value indicating whether the model has an attribute with the specified name.
  137. * @param string $name the name of the attribute.
  138. * @return bool whether the model has an attribute with the specified name.
  139. * @since 2.0.16
  140. */
  141. public function hasAttribute($name)
  142. {
  143. return array_key_exists($name, $this->_attributes);
  144. }
  145. /**
  146. * Defines an attribute.
  147. * @param string $name the attribute name.
  148. * @param mixed $value the attribute value.
  149. */
  150. public function defineAttribute($name, $value = null)
  151. {
  152. $this->_attributes[$name] = $value;
  153. }
  154. /**
  155. * Undefines an attribute.
  156. * @param string $name the attribute name.
  157. */
  158. public function undefineAttribute($name)
  159. {
  160. unset($this->_attributes[$name]);
  161. }
  162. /**
  163. * Adds a validation rule to this model.
  164. * You can also directly manipulate [[validators]] to add or remove validation rules.
  165. * This method provides a shortcut.
  166. * @param string|array $attributes the attribute(s) to be validated by the rule.
  167. * @param string|Validator|\Closure $validator the validator. This can be either:
  168. * * a built-in validator name listed in [[builtInValidators]];
  169. * * a method name of the model class;
  170. * * an anonymous function;
  171. * * a validator class name.
  172. * * a Validator.
  173. * @param array $options the options (name-value pairs) to be applied to the validator.
  174. * @return $this
  175. */
  176. public function addRule($attributes, $validator, $options = [])
  177. {
  178. $validators = $this->getValidators();
  179. if ($validator instanceof Validator) {
  180. $validator->attributes = (array)$attributes;
  181. } else {
  182. $validator = Validator::createValidator($validator, $this, (array)$attributes, $options);
  183. }
  184. $validators->append($validator);
  185. return $this;
  186. }
  187. /**
  188. * Validates the given data with the specified validation rules.
  189. * This method will create a DynamicModel instance, populate it with the data to be validated,
  190. * create the specified validation rules, and then validate the data using these rules.
  191. * @param array $data the data (name-value pairs) to be validated.
  192. * @param array $rules the validation rules. Please refer to [[Model::rules()]] on the format of this parameter.
  193. * @return static the model instance that contains the data being validated.
  194. * @throws InvalidConfigException if a validation rule is not specified correctly.
  195. */
  196. public static function validateData(array $data, $rules = [])
  197. {
  198. /* @var $model DynamicModel */
  199. $model = new static($data);
  200. if (!empty($rules)) {
  201. $validators = $model->getValidators();
  202. foreach ($rules as $rule) {
  203. if ($rule instanceof Validator) {
  204. $validators->append($rule);
  205. } elseif (is_array($rule) && isset($rule[0], $rule[1])) { // attributes, validator type
  206. $validator = Validator::createValidator($rule[1], $model, (array)$rule[0], array_slice($rule, 2));
  207. $validators->append($validator);
  208. } else {
  209. throw new InvalidConfigException('Invalid validation rule: a rule must specify both attribute names and validator type.');
  210. }
  211. }
  212. }
  213. $model->validate();
  214. return $model;
  215. }
  216. /**
  217. * {@inheritdoc}
  218. */
  219. public function attributes()
  220. {
  221. return array_keys($this->_attributes);
  222. }
  223. /**
  224. * Sets the labels for all attributes.
  225. * @param string[] $labels attribute labels.
  226. * @return $this
  227. * @since 2.0.35
  228. */
  229. public function setAttributeLabels(array $labels = [])
  230. {
  231. $this->_attributeLabels = $labels;
  232. return $this;
  233. }
  234. /**
  235. * Sets a label for a single attribute.
  236. * @param string $attribute attribute name.
  237. * @param string $label attribute label value.
  238. * @return $this
  239. * @since 2.0.35
  240. */
  241. public function setAttributeLabel($attribute, $label)
  242. {
  243. $this->_attributeLabels[$attribute] = $label;
  244. return $this;
  245. }
  246. /**
  247. * {@inheritdoc}
  248. */
  249. public function attributeLabels()
  250. {
  251. return $this->_attributeLabels;
  252. }
  253. }