136 lines
4.2KB

  1. <?php
  2. /**
  3. * @link http://www.yiiframework.com/
  4. * @copyright Copyright (c) 2008 Yii Software LLC
  5. * @license http://www.yiiframework.com/license/
  6. */
  7. namespace yii\web;
  8. use Yii;
  9. use yii\base\InvalidConfigException;
  10. /**
  11. * GroupUrlRule represents a collection of URL rules sharing the same prefix in their patterns and routes.
  12. *
  13. * GroupUrlRule is best used by a module which often uses module ID as the prefix for the URL rules.
  14. * For example, the following code creates a rule for the `admin` module:
  15. *
  16. * ```php
  17. * new GroupUrlRule([
  18. * 'prefix' => 'admin',
  19. * 'rules' => [
  20. * 'login' => 'user/login',
  21. * 'logout' => 'user/logout',
  22. * 'dashboard' => 'default/dashboard',
  23. * ],
  24. * ]);
  25. *
  26. * // the above rule is equivalent to the following three rules:
  27. *
  28. * [
  29. * 'admin/login' => 'admin/user/login',
  30. * 'admin/logout' => 'admin/user/logout',
  31. * 'admin/dashboard' => 'admin/default/dashboard',
  32. * ]
  33. * ```
  34. *
  35. * The above example assumes the prefix for patterns and routes are the same. They can be made different
  36. * by configuring [[prefix]] and [[routePrefix]] separately.
  37. *
  38. * Using a GroupUrlRule is more efficient than directly declaring the individual rules it contains.
  39. * This is because GroupUrlRule can quickly determine if it should process a URL parsing or creation request
  40. * by simply checking if the prefix matches.
  41. *
  42. * @author Qiang Xue <qiang.xue@gmail.com>
  43. * @since 2.0
  44. */
  45. class GroupUrlRule extends CompositeUrlRule
  46. {
  47. /**
  48. * @var array the rules contained within this composite rule. Please refer to [[UrlManager::rules]]
  49. * for the format of this property.
  50. * @see prefix
  51. * @see routePrefix
  52. */
  53. public $rules = [];
  54. /**
  55. * @var string the prefix for the pattern part of every rule declared in [[rules]].
  56. * The prefix and the pattern will be separated with a slash.
  57. */
  58. public $prefix;
  59. /**
  60. * @var string the prefix for the route part of every rule declared in [[rules]].
  61. * The prefix and the route will be separated with a slash.
  62. * If this property is not set, it will take the value of [[prefix]].
  63. */
  64. public $routePrefix;
  65. /**
  66. * @var array the default configuration of URL rules. Individual rule configurations
  67. * specified via [[rules]] will take precedence when the same property of the rule is configured.
  68. */
  69. public $ruleConfig = ['class' => 'yii\web\UrlRule'];
  70. /**
  71. * @inheritdoc
  72. */
  73. public function init()
  74. {
  75. $this->prefix = trim($this->prefix, '/');
  76. $this->routePrefix = $this->routePrefix === null ? $this->prefix : trim($this->routePrefix, '/');
  77. parent::init();
  78. }
  79. /**
  80. * @inheritdoc
  81. */
  82. protected function createRules()
  83. {
  84. $rules = [];
  85. foreach ($this->rules as $key => $rule) {
  86. if (!is_array($rule)) {
  87. $rule = [
  88. 'pattern' => ltrim($this->prefix . '/' . $key, '/'),
  89. 'route' => ltrim($this->routePrefix . '/' . $rule, '/'),
  90. ];
  91. } elseif (isset($rule['pattern'], $rule['route'])) {
  92. $rule['pattern'] = ltrim($this->prefix . '/' . $rule['pattern'], '/');
  93. $rule['route'] = ltrim($this->routePrefix . '/' . $rule['route'], '/');
  94. }
  95. $rule = Yii::createObject(array_merge($this->ruleConfig, $rule));
  96. if (!$rule instanceof UrlRuleInterface) {
  97. throw new InvalidConfigException('URL rule class must implement UrlRuleInterface.');
  98. }
  99. $rules[] = $rule;
  100. }
  101. return $rules;
  102. }
  103. /**
  104. * @inheritdoc
  105. */
  106. public function parseRequest($manager, $request)
  107. {
  108. $pathInfo = $request->getPathInfo();
  109. if ($this->prefix === '' || strpos($pathInfo . '/', $this->prefix . '/') === 0) {
  110. return parent::parseRequest($manager, $request);
  111. } else {
  112. return false;
  113. }
  114. }
  115. /**
  116. * @inheritdoc
  117. */
  118. public function createUrl($manager, $route, $params)
  119. {
  120. if ($this->routePrefix === '' || strpos($route, $this->routePrefix . '/') === 0) {
  121. return parent::createUrl($manager, $route, $params);
  122. } else {
  123. return false;
  124. }
  125. }
  126. }