云客Drupal源码分析之主题处理器theme handler

2024-06-09 08:18

本文主要是介绍云客Drupal源码分析之主题处理器theme handler,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

主题处理器theme_handler与主题管理器theme.manager都是系统提供的服务,虽然一字之差但她们功能完全不同,主题管理器用于主题渲染和派发主题修改钩子,而主题处理器完成更为底层的工作,向上为主题管理器、初始化器等各主题相关组件提供基础功能支持,定义如下:
服务id:theme_handler
类:Drupal\Core\Extension\ThemeHandler
获取方法:\Drupal::service("theme_handler");
主要完成的工作是扫描查找文件系统中的主题扩展、理顺继承关系、添加主题info文件的默认项、初始化一些全局函数的静态变量值、判明主题是否应该显示在UI中等等;

主题处理器主要工作是向上层应用提供主题的扩展对象(该对象见下),上层组件有了扩展对象后就可以操作主题了。

主题相关名字:
以下是一些和主题相关的名字及其含义:
机器名:
又叫主题扩展名,全局唯一,是识别主题的唯一标识,来源于info文件的文件名,如“bartik.info.yml”那么机器名就是“bartik”,不能包含空格,正则为:/^[a-zA-Z_\x7f-\xff][a-zA-Z0-9_\x7f-\xff]*$/,有些特定名字虽然满足该正则,但不能被使用,如“template”;下文中除非特别说明,当提到主题名时均指机器名。
人类可读名:
在info文件中name项指定的名称,这可以和机器名不同也可不同,显示在UI中,能包括空格
目录名:
存放主题相关文件的目录名称,注意:主题的扫描查找实际上是在查找内容满足要求的以“.info.yml”后缀结束的文件(内容中必须有type属性),而不是依据目录结构信息,因此主题的目录名可以和机器名不同,但最佳实践应该相同,目录深度与结构不限,目录名不能使用:'src','lib','vendor','assets','css','files','images','js','misc','templates','includes','fixtures','Drupal',
这些目录在被扫描时会被跳过(可以利用这种特性隐藏多余的主题,不会对系统造成任何影响)
主扩展文件名:

默认为“机器名.扩展类型”,如“bartik.theme”、“search.module”,主扩展文件不是必须的,用于存放模块或主题需要的函数或定义常量,位于扩展根目录,是一个php文件,但并不以php做文件名后缀,如果存在则被默认加载。引擎主题是一个特殊情况,命名为“引擎名.engine”,如“twig.engine”。

扩展对象:
当扫描到一个合法的扩展时,系统会初始化一个扩展对象:
\Drupal\Core\Extension\Extension
该对象虽然在类定义中只有很少的属性,但却在初始化后从对象外部赋值了许多属性,见下,这些属性是公共的(public),该对象携带了扩展本身的所有信息,她是主题信息的承载者,初始化时:
$theme = new Extension($root, $type, $pathname, $filename);
四个参数分别是:
$root:drupal根路径,如“C:\root\drupal”没有后缀反斜杠
$type:扩展类型,来自info文件里面的type值
$pathname:info文件相对系统根目录的相对路径,如“core/themes/bartik/bartik.info.yml”
$filename:主扩展文件名,见前文描述,如“bartik.theme”,注意并非info文件的文件名

在主题扩展中常用的扩展对象属性如下(这里以$theme指代扩展对象):
$theme->type:扩展类型,主题是“theme”
$theme->info:从info文件里面解析出来的内容,附加了默认值
$theme->engine:该主题使用的模板引擎,默认为“Twig”
$theme->base_theme:直接基主题的机器名
$theme->libraries:info文件中声明的库
$theme->status:表示主题是否被安装,整数值,1为安装,0为未安装
$theme->owner:所用引擎的主扩展名,通常值为:“core/themes/engines/twig/twig.engine”
$theme->prefix:所用引擎前缀,通常为“twig”
$theme->base_themes:是一个数组,键名是基主题机器名,键值为基主题的人类可读名,如果指定了基主题但其不存在,那么值为NULL,元素顺序代表继承顺序,第一个元素是根基主题,最后一个元素是直接基主题
$theme->base_theme:是一个字符串值,直接基主题机器名,如果没有基主题则该属性不存在
$theme->sub_themes:是一个数组,代表所有子主题,键名是子主题机器名,键值为子主题人类可读名,元素顺序没有意义
$theme->required_by:数组,元素为全部需要该主题的主题(子主题)
$theme->requires:数组,该主题需要的主题(基主题)
$theme->origin:内部为排序目的使用,用于跟踪扩展扫描的起始目录
$theme->subpath:内部为排序目的使用,加上$theme->origin就是相对于系统根目录的子目录
$theme->sort:显示主题时的排序值
$theme->root:系统根路径,如“C:\root\drupal”
$theme->pathname:info文件相对系统根目录的相对路径,如“core/themes/bartik/bartik.info.yml”
$theme->$filename:主扩展文件名,值如“bartik.theme”,注意并非info文件的文件名

该扩展对象中,除类定义的方法外,还可以调用info文件初始化的\SplFileInfo对象上的方法
下文提到扩展对象时均是指该对象。

主题关键配置项与状态值:
在了解主题处理器前我们应该先知道和主题相关的主要配置和信息储存。
\Drupal::service("config.factory")->get('system.theme')->get(“admin”);
储存后台管理主题的机器名
\Drupal::service("config.factory")->get('system.theme')->get(“default”);
储存前端默认主题的机器名
\Drupal::service("config.factory")->get('core.extension')->get('theme');
储存所有被安装的主题,数组值,键名为主题机器名,键值为0
\Drupal::state()->get("system.theme.data");
储存所有被安装的主题,数组值,键名为主题机器名,键值为对应的扩展对象
\Drupal::state()->get("system.theme.files");
储存系统所有主题的info文件路径,包括没有启用的主题,数组值,键名为机器名,键值为相对系统根目录的路径,如:core/themes/bartik/bartik.info.yml
\Drupal::state()->get("system.theme_engine.files");
同上,只是储存模板引擎主题的机器名和路径,默认为[twig=>‘core/themes/engines/twig/twig.info.yml’]
\Drupal::cache('bootstrap')->get('system_list')
缓存主题数据,值为两个元素的数组,键名theme的值为\Drupal::state()->get("system.theme.data")的内容,键名filepaths的值为多个['type' => 'theme','name' => ‘主题机器名’,'filepath' => “info文件路径”,]元素构成的数组,如:['type' => 'theme','name' => ‘bartik’,'filepath' => “core/themes/bartik/bartik.info.yml”,]

主题处理器方法简述:
\Drupal::service("theme_handler")->getDefault();
得到前端主题机器名
setDefault($name);
设置前端主题,参数为机器名,必须是已经安装过的主题,她不会失效如何缓存,所以该方法只应该系统使用内部使用
listInfo()
列出所有已经安装的主题,返回数组值,键名为主题机器名,键值为对应的扩展对象,以扩展目录名作为排序依据(见\Drupal\Core\Extension\ExtensionDiscovery::sort),在很多使用情况下这个顺序是无意义的
addTheme(Extension $theme)
添加一个扩展对象到已安装主题列表
refreshInfo()
重新扫描文件系统,重新构建已经安装主题列表
rebuildThemeData()
扫描文件系统,返回所有存在的主题,包括未安装的主题,理顺主题的继承关系以及设置相关属性变量,返回一个数组,键名为主题机器名,键值为对应的扩展对象。
getBaseThemes(array $themes, $theme)
在一个主题集中查找一个主题的所有基主题,返回一个数组,键名是基主题机器名,键值为基主题的人类可读名,如果指定了基主题但其不存在,那么值为NULL,元素顺序代表继承顺序,第一个元素是根基主题,最后一个元素是直接基主题。
getName($theme)
根据一个主题的机器名返回人类可读名
getThemeDirectories()
得到所有已经安装主题的绝对路径,返回数组,键名为机器名,键值为路径名(不带文件名),如:C:\root\drupal/core/themes/bartik
themeExists($theme)
判断主题是否存在于已安装主题中
getTheme($name)
通过主题机器名取得扩展对象
hasUi($name)
判断主题是否应该在ui中出现,如果info信息中'hidden'为真,那么不出现,如果省略该项设置,默认出现,如果是默认的管理主题或前端主题,那么强制出现

此外安装卸载方法虽然在8.x版本中存在,但你不应该使用,将在9.0版本中移除,使用安装服务代替

主题info文件的附加默认值:
主题info文件被系统读取时会附加默认值,采用数组“+”操作,由此可见在没有指定选项时她会是什么样子,默认值被定义在主题处理器中,如下:

$defaults = ['engine' => 'twig','base theme' => 'stable','regions' => ['sidebar_first' => 'Left sidebar','sidebar_second' => 'Right sidebar','content' => 'Content','header' => 'Header','primary_menu' => 'Primary menu','secondary_menu' => 'Secondary menu','footer' => 'Footer','highlighted' => 'Highlighted','help' => 'Help','page_top' => 'Page top','page_bottom' => 'Page bottom','breadcrumb' => 'Breadcrumb',],'description' => '','features' =>  ['favicon','logo','node_user_picture','comment_user_picture','comment_user_verification',],'screenshot' => 'screenshot.png','php' => DRUPAL_MINIMUM_PHP,'libraries' => [],
];

由于采用数组后附加操作,所以当不想要某默认值时,需要自定义该值,比如不想要基主题那么需要明确定义为FALSE,否则默认为'stable'主题。
默认值被附加后,会派发system_info修改钩:
$moduleHandler->alter('system_info', $theme->info, $theme, $type);
这里 $theme是前文提到的\Drupal\Core\Extension\Extension对象,info是对象外附加的公共属性,内容是应用了默认值的info文件内容数组,$type值为'theme',模块可以通过hook_system_info_alter()去修改她,接收三个参数,第一个和第三个参数实际上是包含在Extension对象中的,由于php对象默认是按引用传递所以模块可以修改该对象(修改$info数组参数将无作用,她不是引用传递),由于派发钩子的原因,意味着先安装的模块可以修改后面正在安装主题的info文件

主题的继承:
可以一层一层的继承下去,层数不限,但只有一个直接基主题(父主题),也就是“单基多层继承”,这和php类继承类似。
drupal是可以使用多个主题引擎的,如果在主题继承链中,子主题和基主题指定了不同的主题引擎,那么将以根基主题的设置为准。
当声明了基主题后,就自动声明了基主题依赖,在info文件中不必手动指定

补充:
1、    在info文件里面注意空格的问题,如“'base theme'”项写成了“'base  theme'”将失去作用,那影响了数组键名,在建立info文件时最好复制修改,避免这些小问题。
2、    在主题处理器中使用了一些编程技巧加快系统性能,如:为主题初始化drupal_get_filename全局函数,通过drupal_static函数设置中心静态属性等,有兴趣可以自行深究
 

我是云客,【云游天下,做客四方】,联系方式见主页,欢迎转载,但须注明出处

 

 

 


 

 

这篇关于云客Drupal源码分析之主题处理器theme handler的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1044645

相关文章

关于MyISAM和InnoDB对比分析

《关于MyISAM和InnoDB对比分析》:本文主要介绍关于MyISAM和InnoDB对比分析,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录开篇:从交通规则看存储引擎选择理解存储引擎的基本概念技术原理对比1. 事务支持:ACID的守护者2. 锁机制:并发控制的艺

苹果macOS 26 Tahoe主题功能大升级:可定制图标/高亮文本/文件夹颜色

《苹果macOS26Tahoe主题功能大升级:可定制图标/高亮文本/文件夹颜色》在整体系统设计方面,macOS26采用了全新的玻璃质感视觉风格,应用于Dock栏、应用图标以及桌面小部件等多个界面... 科技媒体 MACRumors 昨日(6 月 13 日)发布博文,报道称在 macOS 26 Tahoe 中

MyBatis Plus 中 update_time 字段自动填充失效的原因分析及解决方案(最新整理)

《MyBatisPlus中update_time字段自动填充失效的原因分析及解决方案(最新整理)》在使用MyBatisPlus时,通常我们会在数据库表中设置create_time和update... 目录前言一、问题现象二、原因分析三、总结:常见原因与解决方法对照表四、推荐写法前言在使用 MyBATis

Python主动抛出异常的各种用法和场景分析

《Python主动抛出异常的各种用法和场景分析》在Python中,我们不仅可以捕获和处理异常,还可以主动抛出异常,也就是以类的方式自定义错误的类型和提示信息,这在编程中非常有用,下面我将详细解释主动抛... 目录一、为什么要主动抛出异常?二、基本语法:raise关键字基本示例三、raise的多种用法1. 抛

github打不开的问题分析及解决

《github打不开的问题分析及解决》:本文主要介绍github打不开的问题分析及解决,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、找到github.com域名解析的ip地址二、找到github.global.ssl.fastly.net网址解析的ip地址三

Mysql的主从同步/复制的原理分析

《Mysql的主从同步/复制的原理分析》:本文主要介绍Mysql的主从同步/复制的原理分析,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录为什么要主从同步?mysql主从同步架构有哪些?Mysql主从复制的原理/整体流程级联复制架构为什么好?Mysql主从复制注意

java -jar命令运行 jar包时运行外部依赖jar包的场景分析

《java-jar命令运行jar包时运行外部依赖jar包的场景分析》:本文主要介绍java-jar命令运行jar包时运行外部依赖jar包的场景分析,本文给大家介绍的非常详细,对大家的学习或工作... 目录Java -jar命令运行 jar包时如何运行外部依赖jar包场景:解决:方法一、启动参数添加: -Xb

Apache 高级配置实战之从连接保持到日志分析的完整指南

《Apache高级配置实战之从连接保持到日志分析的完整指南》本文带你从连接保持优化开始,一路走到访问控制和日志管理,最后用AWStats来分析网站数据,对Apache配置日志分析相关知识感兴趣的朋友... 目录Apache 高级配置实战:从连接保持到日志分析的完整指南前言 一、Apache 连接保持 - 性

Linux中的more 和 less区别对比分析

《Linux中的more和less区别对比分析》在Linux/Unix系统中,more和less都是用于分页查看文本文件的命令,但less是more的增强版,功能更强大,:本文主要介绍Linu... 目录1. 基础功能对比2. 常用操作对比less 的操作3. 实际使用示例4. 为什么推荐 less?5.

spring-gateway filters添加自定义过滤器实现流程分析(可插拔)

《spring-gatewayfilters添加自定义过滤器实现流程分析(可插拔)》:本文主要介绍spring-gatewayfilters添加自定义过滤器实现流程分析(可插拔),本文通过实例图... 目录需求背景需求拆解设计流程及作用域逻辑处理代码逻辑需求背景公司要求,通过公司网络代理访问的请求需要做请