как Yii2 заменяет @app/views/layouts/main.php с @app/themes/basic/layouts/main.php когда я использую компонент просмотра
Если я использую компонент просмотра, как сказано в документе
return [
'components' => [
'view' => [
'theme' => [
'basePath' => '@app/themes/basic',
'baseUrl' => '@web/themes/basic',
'pathMap' => [
'@app/views' => '@app/themes/basic',
],
],
],
],
];
Означает ли это, что main.php в папке @app/themes/basic/layouts/main.php заменит main.php в папке @app/views/layouts/main.php ?
И как yii2 этого добивается? Я не мог понять, хотя я проследил за контроллером, чтобы найти что-то, что могло бы помочь мне понять логику.
Я проследил коды следующим образом, но я не выяснил, как макет main.php был заменен, когда использование тем.
1. Контроллер сайта, индекс действий по умолчанию, затем визуализируйте представление
public function actionIndex()
{
return $this->render('index');
}
2. вызовите функцию класса контроллера render
public function render($view, $params = [])
{
$content = $this->getView()->render($view, $params, $this);
return $this->renderContent($content);
}
3. получите представление yii\web\
public function getView()
{
if ($this->_view === null) {
$this->_view = Yii::$app->getView();
}
return $this->_view;
}
4. используйте функцию класса представления render
public function render($view, $params = [], $context = null)
{
$viewFile = $this->findViewFile($view, $context);
return $this->renderFile($viewFile, $params, $context);
}
5. найдите файл представления, найдите путь к файлу представления
protected function findViewFile($view, $context = null)
{
if (strncmp($view, '@', 1) === 0) {
// e.g. "@app/views/main"
$file = Yii::getAlias($view);
} elseif (strncmp($view, '//', 2) === 0) {
// e.g. "//layouts/main"
$file = Yii::$app->getViewPath() . DIRECTORY_SEPARATOR . ltrim($view, '/');
} elseif (strncmp($view, '/', 1) === 0) {
// e.g. "/site/index"
if (Yii::$app->controller !== null) {
$file = Yii::$app->controller->module->getViewPath() . DIRECTORY_SEPARATOR . ltrim($view, '/');
} else {
throw new InvalidCallException("Unable to locate view file for view '$view': no active controller.");
}
} elseif ($context instanceof ViewContextInterface) {
$file = $context->getViewPath() . DIRECTORY_SEPARATOR . $view;
} elseif (($currentViewFile = $this->getViewFile()) !== false) {
$file = dirname($currentViewFile) . DIRECTORY_SEPARATOR . $view;
} else {
throw new InvalidCallException("Unable to resolve view file for view '$view': no active view context.");
}
if (pathinfo($file, PATHINFO_EXTENSION) !== '') {
return $file;
}
$path = $file . '.' . $this->defaultExtension;
if ($this->defaultExtension !== 'php' && !is_file($path)) {
$path = $file . '.php';
}
return $path;
}
6. визуализируйте этот файл представления
public function renderFile($viewFile, $params = [], $context = null)
{
$viewFile = Yii::getAlias($viewFile);
if ($this->theme !== null) {
$viewFile = $this->theme->applyTo($viewFile);
}
if (is_file($viewFile)) {
$viewFile = FileHelper::localize($viewFile);
} else {
throw new ViewNotFoundException("The view file does not exist: $viewFile");
}
$oldContext = $this->context;
if ($context !== null) {
$this->context = $context;
}
$output = '';
$this->_viewFiles[] = $viewFile;
if ($this->beforeRender($viewFile, $params)) {
Yii::trace("Rendering view file: $viewFile", __METHOD__);
$ext = pathinfo($viewFile, PATHINFO_EXTENSION);
if (isset($this->renderers[$ext])) {
if (is_array($this->renderers[$ext]) || is_string($this->renderers[$ext])) {
$this->renderers[$ext] = Yii::createObject($this->renderers[$ext]);
}
/* @var $renderer ViewRenderer */
$renderer = $this->renderers[$ext];
$output = $renderer->render($this, $viewFile, $params);
} else {
$output = $this->renderPhpFile($viewFile, $params);
}
$this->afterRender($viewFile, $params, $output);
}
array_pop($this->_viewFiles);
$this->context = $oldContext;
return $output;
}
7. если тема не является нулевой, система вызовет команду $this->theme->applyTo($viewfile), чтобы получить файл представления темы, и Файл визуализации
$output = $this->renderPhpFile($viewFile, $params);
8. получите содержимое этого файла
1 answers
Это происходит потому, что, когда Yii2 выполняет проверку рендеринга в конфигурации/компонентах, реальное сопоставление для представления контейнера direrctory.
Это может быть связано со всеми представлениями, как в вашем случае
'view' => [
'theme' => [
'basePath' => '@app/themes/basic',
'baseUrl' => '@web/themes/basic',
'pathMap' => [
'@app/views' => '@app/themes/basic',
],
],
],
Или может быть связано с каким-либо поставщиком или по модулю как
'view' => [
'theme' => [
'pathMap' => [
'@dektrium/user/views' => '@backend/views/my-view-user' // mapping per overriding s dektrium views with personal views
],
],
],
Вы можете взглянуть на справочный документ для просмотра http://www.yiiframework.com/doc-2.0/yii-base-view.html
И https://github.com/yiisoft/yii2/blob/master/framework/base/View.php