flack /
openpsa
| 1 | <?php |
||
| 2 | /** |
||
| 3 | * @package midcom.templating |
||
| 4 | * @author CONTENT CONTROL http://www.contentcontrol-berlin.de/ |
||
| 5 | * @copyright CONTENT CONTROL http://www.contentcontrol-berlin.de/ |
||
| 6 | * @license http://www.gnu.org/licenses/gpl.html GNU General Public License |
||
| 7 | */ |
||
| 8 | |||
| 9 | namespace midcom\templating; |
||
| 10 | |||
| 11 | use midcom; |
||
| 12 | use midcom_db_topic; |
||
| 13 | use midcom_core_context; |
||
| 14 | use midcom_connection; |
||
| 15 | |||
| 16 | /** |
||
| 17 | * templating loader class |
||
| 18 | * |
||
| 19 | * Style Inheritance |
||
| 20 | * |
||
| 21 | * The basic path the styleloader follows to find a style element is: |
||
| 22 | * 1. Topic style -> if the current topic has a style set |
||
| 23 | * 2. Inherited topic style -> if the topic inherits a style from another topic. |
||
| 24 | * 3. Site-wide per-component default style -> if defined in MidCOM configuration key styleengine_default_styles |
||
| 25 | * 4. Theme style -> the style of the MidCOM component. |
||
| 26 | * 5. The file style. This is usually the elements found in the component's style directory. |
||
| 27 | * |
||
| 28 | * Regarding no. 5: |
||
| 29 | * It is possible to add extra file styles if so is needed for example by a portal component. |
||
| 30 | * This is done either using the append/prepend component_style functions of midcom::get()->style or by setting it |
||
| 31 | * to another directory by calling (append|prepend)_styledir directly. |
||
| 32 | * |
||
| 33 | * @package midcom.templating |
||
| 34 | */ |
||
| 35 | class loader |
||
| 36 | { |
||
| 37 | /** |
||
| 38 | * Default style element cache |
||
| 39 | * |
||
| 40 | * @var string[] |
||
| 41 | */ |
||
| 42 | protected array $cache = []; |
||
| 43 | |||
| 44 | /** |
||
| 45 | * The stack of directories to check for styles. |
||
| 46 | * |
||
| 47 | * @var string[] |
||
| 48 | */ |
||
| 49 | private array $directories = []; |
||
| 50 | |||
| 51 | private string $theme_root; |
||
| 52 | |||
| 53 | 1 | public function __construct(string $theme_root) |
|
| 54 | { |
||
| 55 | 1 | $this->theme_root = $theme_root; |
|
| 56 | } |
||
| 57 | |||
| 58 | 277 | public function set_directories(?midcom_db_topic $topic, array $prepend, array $append) |
|
| 59 | { |
||
| 60 | 277 | $this->directories = $prepend; |
|
| 61 | 277 | if ($snippetdir = $this->get_component_snippetdir($topic)) { |
|
| 62 | 277 | $this->directories[] = $snippetdir; |
|
| 63 | } |
||
| 64 | 277 | $this->directories = array_merge($this->directories, $append); |
|
| 65 | } |
||
| 66 | |||
| 67 | /** |
||
| 68 | * Gets the component styledir associated with the topic's component. |
||
| 69 | * |
||
| 70 | * @return ?string the path to the component's style directory. |
||
| 71 | */ |
||
| 72 | 277 | private function get_component_snippetdir(?midcom_db_topic $topic) : ?string |
|
| 73 | { |
||
| 74 | 277 | if (empty($topic->component)) { |
|
| 75 | 2 | return null; |
|
| 76 | } |
||
| 77 | 277 | return midcom::get()->componentloader->path_to_snippetpath($topic->component) . "/style"; |
|
| 78 | } |
||
| 79 | |||
| 80 | /** |
||
| 81 | * Try to get element from default style snippet |
||
| 82 | */ |
||
| 83 | 207 | public function get_element(string $name, bool $scope_from_path) : ?string |
|
| 84 | { |
||
| 85 | 207 | if (midcom::get()->config->get('theme')) { |
|
| 86 | 207 | $src = "theme:{$name}"; |
|
| 87 | 207 | if (array_key_exists($src, $this->cache)) { |
|
| 88 | 15 | return $this->cache[$src]; |
|
| 89 | } |
||
| 90 | 207 | if ($content = $this->get_element_from_theme($name)) { |
|
| 91 | 2 | return $this->add_to_cache($src, $content); |
|
| 92 | } |
||
| 93 | } |
||
| 94 | |||
| 95 | 207 | foreach ($this->directories as $path) { |
|
| 96 | 207 | $filename = $path . "/{$name}.php"; |
|
| 97 | 207 | if (file_exists($filename)) { |
|
| 98 | 197 | if (array_key_exists($filename, $this->cache)) { |
|
| 99 | 78 | return $this->cache[$filename]; |
|
| 100 | } |
||
| 101 | 150 | return $this->add_to_cache($filename, file_get_contents($filename)); |
|
| 102 | } |
||
| 103 | } |
||
| 104 | 24 | return null; |
|
| 105 | } |
||
| 106 | |||
| 107 | /** |
||
| 108 | * Get the content of the element by the passed element name. |
||
| 109 | * Tries to resolve path according to theme-name & page |
||
| 110 | */ |
||
| 111 | 207 | private function get_element_from_theme(string $element_name) : ?string |
|
| 112 | { |
||
| 113 | 207 | $theme = midcom::get()->config->get('theme'); |
|
| 114 | 207 | $path_array = explode('/', $theme); |
|
|
0 ignored issues
–
show
Bug
introduced
by
Loading history...
|
|||
| 115 | |||
| 116 | //get the page if there is one |
||
| 117 | 207 | $page = midcom_connection::get_url('page_style'); |
|
| 118 | 207 | $substyle = midcom_core_context::get()->get_key(MIDCOM_CONTEXT_SUBSTYLE); |
|
| 119 | //check if we have elements for the sub-styles |
||
| 120 | 207 | while (!empty($path_array)) { |
|
| 121 | 207 | $theme_path = implode('/', $path_array); |
|
| 122 | 207 | $candidates = []; |
|
| 123 | 207 | if ($substyle) { |
|
| 124 | 31 | $candidates[] = '/' . $substyle . '/' . $element_name; |
|
| 125 | } |
||
| 126 | 207 | if ($page) { |
|
| 127 | $candidates[] = $page . '/' . $element_name; |
||
| 128 | } |
||
| 129 | 207 | $candidates[] = '/' . $element_name; |
|
| 130 | |||
| 131 | 207 | foreach ($candidates as $candidate) { |
|
| 132 | 207 | $filename = $this->theme_root . $theme_path . '/style' . $candidate . '.php'; |
|
| 133 | 207 | if (file_exists($filename)) { |
|
| 134 | 2 | return file_get_contents($filename); |
|
| 135 | } |
||
| 136 | } |
||
| 137 | |||
| 138 | //remove last theme part |
||
| 139 | 207 | array_pop($path_array); |
|
| 140 | } |
||
| 141 | |||
| 142 | 207 | return null; |
|
| 143 | } |
||
| 144 | |||
| 145 | 151 | protected function add_to_cache(string $cache_key, string $content) : string |
|
| 146 | { |
||
| 147 | 151 | $this->cache[$cache_key] = $this->resolve_includes($content); |
|
| 148 | 151 | return $this->cache[$cache_key]; |
|
| 149 | } |
||
| 150 | |||
| 151 | 151 | private function resolve_includes(string $content) : string |
|
| 152 | { |
||
| 153 | 151 | return preg_replace_callback("/<\\(([a-zA-Z0-9 _-]+)\\)>/", function (array $matches) { |
|
| 154 | 3 | $element = $matches[1]; |
|
| 155 | |||
| 156 | switch ($element) { |
||
| 157 | 3 | case 'title': |
|
| 158 | 2 | return '<?= midcom::get()->config->get("midcom_site_title") ?>'; |
|
| 159 | 2 | case 'content': |
|
| 160 | 2 | return '<?php midcom_core_context::get()->show(); ?>'; |
|
| 161 | default: |
||
| 162 | 1 | if ($value = $this->get_element_from_theme($element)) { |
|
| 163 | 1 | return $this->resolve_includes($value); |
|
| 164 | } |
||
| 165 | return ''; |
||
| 166 | } |
||
| 167 | 151 | }, $content); |
|
| 168 | } |
||
| 169 | |||
| 170 | /** |
||
| 171 | * Initializes style sources |
||
| 172 | */ |
||
| 173 | 277 | public function initialize(midcom_core_context $context, ?string $style) |
|
| 174 | { |
||
| 175 | 277 | if ($style && str_starts_with($style, 'theme:')) { |
|
| 176 | $theme_dir = $this->theme_root . midcom::get()->config->get('theme') . '/style'; |
||
| 177 | $parts = explode('/', str_replace('theme:/', '', $style)); |
||
| 178 | |||
| 179 | foreach ($parts as &$part) { |
||
| 180 | $theme_dir .= '/' . $part; |
||
| 181 | $part = $theme_dir; |
||
| 182 | } |
||
| 183 | foreach (array_reverse(array_filter($parts, is_dir(...))) as $dirname) { |
||
|
0 ignored issues
–
show
The type
midcom\templating\is_dir was not found. Maybe you did not declare it correctly or list all dependencies?
The issue could also be caused by a filter entry in the build configuration.
If the path has been excluded in your configuration, e.g. filter:
dependency_paths: ["lib/*"]
For further information see https://scrutinizer-ci.com/docs/tools/php/php-scrutinizer/#list-dependency-paths Loading history...
|
|||
| 184 | midcom::get()->style->prepend_styledir($dirname); |
||
| 185 | } |
||
| 186 | } |
||
| 187 | } |
||
| 188 | } |
||
| 189 |