как 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. получите содержимое этого файла

Author: Arron, 2017-08-29

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

 0
Author: scaisEdge, 2017-08-29 06:09:19