Commit 62f43e17 by Philippe Gaultier

Merge branch 'master' into feature/CORS

parents ba74d97a f66e28c4
Components Components
========== ==========
Components are the main building blocks of Yii applications. Components are instances of [[yii\base\Component]] Components are the main building blocks of Yii applications. Components are instances of [[yii\base\Component]],
or an extended class. The three main features that components provide to other classes are: or an extended class. The three main features that components provide to other classes are:
* [Properties](concept-properties.md) * [Properties](concept-properties.md)
...@@ -26,10 +26,10 @@ echo DatePicker::widget([ ...@@ -26,10 +26,10 @@ echo DatePicker::widget([
The widget's properties are easily writable because the class extends [[yii\base\Component]]. The widget's properties are easily writable because the class extends [[yii\base\Component]].
While components are very powerful, they are a bit heavier than normal objects, due to the fact that While components are very powerful, they are a bit heavier than normal objects, due to the fact that
it takes extra memory and CPU time to support [events](concept-events.md) and [behaviors](concept-behaviors.md) in particular. it takes extra memory and CPU time to support [event](concept-events.md) and [behavior](concept-behaviors.md) functionality in particular.
If your components do not need these two features, you may consider extending your component class from If your components do not need these two features, you may consider extending your component class from
[[yii\base\Object]] instead of [[yii\base\Component]]. Doing so will make your components as efficient as normal PHP objects, [[yii\base\Object]] instead of [[yii\base\Component]]. Doing so will make your components as efficient as normal PHP objects,
but with the added support for [properties](concept-properties.md). but with added support for [properties](concept-properties.md).
When extending your class from [[yii\base\Component]] or [[yii\base\Object]], it is recommended that you follow When extending your class from [[yii\base\Component]] or [[yii\base\Object]], it is recommended that you follow
these conventions: these conventions:
...@@ -67,7 +67,7 @@ class MyClass extends Object ...@@ -67,7 +67,7 @@ class MyClass extends Object
} }
``` ```
Following these guideliness will make your components [configurable](concept-configurations.md) when they are created. For example: Following these guideliness will make your components [configurable](concept-configurations.md) when they are instantiated. For example:
```php ```php
$component = new MyClass(1, 2, ['prop1' => 3, 'prop2' => 4]); $component = new MyClass(1, 2, ['prop1' => 3, 'prop2' => 4]);
...@@ -79,8 +79,7 @@ $component = \Yii::createObject([ ...@@ -79,8 +79,7 @@ $component = \Yii::createObject([
], [1, 2]); ], [1, 2]);
``` ```
> Info: While the approach of calling [[Yii::createObject()]] looks more complicated, it is more powerful due to > Info: While the approach of calling [[Yii::createObject()]] looks more complicated, it is more powerful because it is implemented on top of a [dependency injection container](concept-di-container.md).
the fact that it is implemented on top of a [dependency injection container](concept-di-container.md).
The [[yii\base\Object]] class enforces the following object lifecycle: The [[yii\base\Object]] class enforces the following object lifecycle:
...@@ -90,5 +89,5 @@ The [[yii\base\Object]] class enforces the following object lifecycle: ...@@ -90,5 +89,5 @@ The [[yii\base\Object]] class enforces the following object lifecycle:
3. Post-initialization within [[yii\base\Object::init()|init()]]. You may override this method to perform sanity checks and normalization of the properties. 3. Post-initialization within [[yii\base\Object::init()|init()]]. You may override this method to perform sanity checks and normalization of the properties.
4. Object method calls. 4. Object method calls.
The first three steps all happen within the object's constructor. This means that once you get an object instance, The first three steps all happen within the object's constructor. This means that once you get a class instance (i.e., an object),
it has already been initialized to a proper state that you can reliably work with. that object has already been initialized to a proper, reliable state.
...@@ -2,19 +2,19 @@ Properties ...@@ -2,19 +2,19 @@ Properties
========== ==========
In PHP, class member variables are also called *properties*. These variables are part of the class definition, and are used In PHP, class member variables are also called *properties*. These variables are part of the class definition, and are used
to represent the state of a class instance (i.e., to differentiate one instance of the class from another). In practice, you may often want to handle the reading or writing of properties in special ways. For example, you may want to trim a string when it is being assigned to represent the state of a class instance (i.e., to differentiate one instance of the class from another). In practice, you may often want to handle the reading or writing of properties in special ways. For example, you may want to always trim a string when it is being assigned
to a `label` property. You could use the following code to achieve this task: to a `label` property. You *could* use the following code to achieve this task:
```php ```php
$object->label = trim($label); $object->label = trim($label);
``` ```
The drawback of the above code is that you have to call `trim()` everywhere in your code where you might set the `label` The drawback of the above code is that you would have to call `trim()` everywhere in your code where you might set the `label`
property. If in the future, the `label` property gets a new requirement, such as the first letter must be captialized, you would again have to modify every bit of code that assigns a value to `label`. The repetition of code leads to bugs and is a practice you want to avoid as much as possible. property. If, in the future, the `label` property gets a new requirement, such as the first letter must be captialized, you would again have to modify every bit of code that assigns a value to `label`. The repetition of code leads to bugs, and is a practice you want to avoid as much as possible.
To solve this problem, Yii introduces a base class called [[yii\base\Object]] that supports defining properties To solve this problem, Yii introduces a base class called [[yii\base\Object]] that supports defining properties
based on *getter* and *setter* class methods. If a class needs such support, it should extend from based on *getter* and *setter* class methods. If a class needs that functionality, it should extend from
[[yii\base\Object]] or a child class. [[yii\base\Object]], or from a child class.
> Info: Nearly every core class in the Yii framework extends from [[yii\base\Object]] or a child class. > Info: Nearly every core class in the Yii framework extends from [[yii\base\Object]] or a child class.
This means that whenever you see a getter or setter in a core class, you can use it like a property. This means that whenever you see a getter or setter in a core class, you can use it like a property.
...@@ -69,9 +69,9 @@ There are several special rules for, and limitations on, the properties defined ...@@ -69,9 +69,9 @@ There are several special rules for, and limitations on, the properties defined
This is because method names in PHP are case-insensitive. This is because method names in PHP are case-insensitive.
* If the name of such a property is the same as a class member variable, the latter will take precedence. * If the name of such a property is the same as a class member variable, the latter will take precedence.
For example, if the above `Foo` class has a member variable `label`, then the assignment `$object->label = 'abc'` For example, if the above `Foo` class has a member variable `label`, then the assignment `$object->label = 'abc'`
will affect the member variable 'label', that line would not call the `setLabel()` setter method. will affect the *member variable* 'label'; that line would not call the `setLabel()` setter method.
* These properties do not support visibility. It makes no difference for the visibility of a property * These properties do not support visibility. It makes no difference for the visibility of a property
if the defining getter or setter method is public, protected or private. if the defining getter or setter method is public, protected or private.
* The properties can only be defined by *non-static* getters and/or setters. Static methods will not be treated in this same manner. * The properties can only be defined by *non-static* getters and/or setters. Static methods will not be treated in this same manner.
Returning back to the problem described at the beginning of this guide, instead of calling `trim()` everywhere a `label` value is assigned, `trim()` only needs to be invoked within the setter `setLabel()`. And if a new requirement comes that requires the label be initially capitalized, the `setLabel()` method can quickly be modified without touching any other code. The one change will universally affect every assignment to `label`. Returning back to the problem described at the beginning of this guide, instead of calling `trim()` everywhere a `label` value is assigned, `trim()` now only needs to be invoked within the setter `setLabel()`. And if a new requirement comes that requires the label be initially capitalized, the `setLabel()` method can quickly be modified without touching any other code. The one change will universally affect every assignment to `label`.
...@@ -71,6 +71,7 @@ Yii Framework 2 Change Log ...@@ -71,6 +71,7 @@ Yii Framework 2 Change Log
- Bug: Fixed Object of class Imagick could not be converted to string in CaptchaAction (eXprojects, cebe) - Bug: Fixed Object of class Imagick could not be converted to string in CaptchaAction (eXprojects, cebe)
- Enh #87: Helper `yii\helpers\Security` converted into application component, cryptographic strength improved (klimov-paul) - Enh #87: Helper `yii\helpers\Security` converted into application component, cryptographic strength improved (klimov-paul)
- Enh #422: Added Support for BIT(M) data type default values in Schema (cebe) - Enh #422: Added Support for BIT(M) data type default values in Schema (cebe)
- Enh #1160: Added $strict parameter to Inflector::camel2id() to handle consecutive uppercase chars (schmunk)
- Enh #1452: Added `Module::getInstance()` to allow accessing the module instance from anywhere within the module (qiangxue) - Enh #1452: Added `Module::getInstance()` to allow accessing the module instance from anywhere within the module (qiangxue)
- Enh #2264: `CookieCollection::has()` will return false for expired or removed cookies (qiangxue) - Enh #2264: `CookieCollection::has()` will return false for expired or removed cookies (qiangxue)
- Enh #2435: `yii\db\IntegrityException` is now thrown on database integrity errors instead of general `yii\db\Exception` (samdark) - Enh #2435: `yii\db\IntegrityException` is now thrown on database integrity errors instead of general `yii\db\Exception` (samdark)
......
...@@ -77,9 +77,9 @@ class RateLimiter extends ActionFilter ...@@ -77,9 +77,9 @@ class RateLimiter extends ActionFilter
$action $action
); );
} elseif ($user) { } elseif ($user) {
Yii::info('Rate limit skipped: "user" does not implement RateLimitInterface.'); Yii::info('Rate limit skipped: "user" does not implement RateLimitInterface.', __METHOD__);
} else { } else {
Yii::info('Rate limit skipped: user not logged in.'); Yii::info('Rate limit skipped: user not logged in.', __METHOD__);
} }
return true; return true;
} }
......
...@@ -326,14 +326,16 @@ class BaseInflector ...@@ -326,14 +326,16 @@ class BaseInflector
* For example, 'PostTag' will be converted to 'post-tag'. * For example, 'PostTag' will be converted to 'post-tag'.
* @param string $name the string to be converted * @param string $name the string to be converted
* @param string $separator the character used to concatenate the words in the ID * @param string $separator the character used to concatenate the words in the ID
* @param string $strict whether to insert a separator between two consecutive uppercase chars, defaults to false
* @return string the resulting ID * @return string the resulting ID
*/ */
public static function camel2id($name, $separator = '-') public static function camel2id($name, $separator = '-', $strict = false)
{ {
$regex = $strict ? '/[A-Z]/' : '/(?<![A-Z])[A-Z]/';
if ($separator === '_') { if ($separator === '_') {
return trim(strtolower(preg_replace('/(?<![A-Z])[A-Z]/', '_\0', $name)), '_'); return trim(strtolower(preg_replace($regex, '_\0', $name)), '_');
} else { } else {
return trim(strtolower(str_replace('_', $separator, preg_replace('/(?<![A-Z])[A-Z]/', $separator . '\0', $name))), $separator); return trim(strtolower(str_replace('_', $separator, preg_replace($regex, $separator . '\0', $name))), $separator);
} }
} }
......
...@@ -113,7 +113,7 @@ class I18N extends Component ...@@ -113,7 +113,7 @@ class I18N extends Component
$result = $formatter->format($message, $params, $language); $result = $formatter->format($message, $params, $language);
if ($result === false) { if ($result === false) {
$errorMessage = $formatter->getErrorMessage(); $errorMessage = $formatter->getErrorMessage();
Yii::warning("Formatting message for language '$language' failed with error: $errorMessage. The message being formatted was: $message."); Yii::warning("Formatting message for language '$language' failed with error: $errorMessage. The message being formatted was: $message.", __METHOD__);
return $message; return $message;
} else { } else {
......
...@@ -94,7 +94,7 @@ class AssetConverter extends Component implements AssetConverterInterface ...@@ -94,7 +94,7 @@ class AssetConverter extends Component implements AssetConverterInterface
} elseif (YII_DEBUG) { } elseif (YII_DEBUG) {
throw new Exception("AssetConverter command '$command' failed with exit code $status:\nSTDOUT:\n$stdout\nSTDERR:\n$stderr"); throw new Exception("AssetConverter command '$command' failed with exit code $status:\nSTDOUT:\n$stdout\nSTDERR:\n$stderr");
} else { } else {
Yii::error("AssetConverter command '$command' failed with exit code $status:\nSTDOUT:\n$stdout\nSTDERR:\n$stderr"); Yii::error("AssetConverter command '$command' failed with exit code $status:\nSTDOUT:\n$stdout\nSTDERR:\n$stderr", __METHOD__);
} }
return $status === 0; return $status === 0;
......
...@@ -95,6 +95,12 @@ class InflectorTest extends TestCase ...@@ -95,6 +95,12 @@ class InflectorTest extends TestCase
$this->assertEquals('post-tag', Inflector::camel2id('postTag')); $this->assertEquals('post-tag', Inflector::camel2id('postTag'));
$this->assertEquals('post_tag', Inflector::camel2id('postTag', '_')); $this->assertEquals('post_tag', Inflector::camel2id('postTag', '_'));
$this->assertEquals('foo-ybar', Inflector::camel2id('FooYBar', '-', false));
$this->assertEquals('foo_ybar', Inflector::camel2id('fooYBar', '_', false));
$this->assertEquals('foo-y-bar', Inflector::camel2id('FooYBar', '-', true));
$this->assertEquals('foo_y_bar', Inflector::camel2id('fooYBar', '_', true));
} }
public function testId2camel() public function testId2camel()
...@@ -104,6 +110,9 @@ class InflectorTest extends TestCase ...@@ -104,6 +110,9 @@ class InflectorTest extends TestCase
$this->assertEquals('PostTag', Inflector::id2camel('post-tag')); $this->assertEquals('PostTag', Inflector::id2camel('post-tag'));
$this->assertEquals('PostTag', Inflector::id2camel('post_tag', '_')); $this->assertEquals('PostTag', Inflector::id2camel('post_tag', '_'));
$this->assertEquals('FooYBar', Inflector::id2camel('foo-y-bar'));
$this->assertEquals('FooYBar', Inflector::id2camel('foo_y_bar', '_'));
} }
public function testHumanize() public function testHumanize()
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment