云客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

相关文章

Java汇编源码如何查看环境搭建

《Java汇编源码如何查看环境搭建》:本文主要介绍如何在IntelliJIDEA开发环境中搭建字节码和汇编环境,以便更好地进行代码调优和JVM学习,首先,介绍了如何配置IntelliJIDEA以方... 目录一、简介二、在IDEA开发环境中搭建汇编环境2.1 在IDEA中搭建字节码查看环境2.1.1 搭建步

Redis主从复制实现原理分析

《Redis主从复制实现原理分析》Redis主从复制通过Sync和CommandPropagate阶段实现数据同步,2.8版本后引入Psync指令,根据复制偏移量进行全量或部分同步,优化了数据传输效率... 目录Redis主DodMIK从复制实现原理实现原理Psync: 2.8版本后总结Redis主从复制实

锐捷和腾达哪个好? 两个品牌路由器对比分析

《锐捷和腾达哪个好?两个品牌路由器对比分析》在选择路由器时,Tenda和锐捷都是备受关注的品牌,各自有独特的产品特点和市场定位,选择哪个品牌的路由器更合适,实际上取决于你的具体需求和使用场景,我们从... 在选购路由器时,锐捷和腾达都是市场上备受关注的品牌,但它们的定位和特点却有所不同。锐捷更偏向企业级和专

Spring中Bean有关NullPointerException异常的原因分析

《Spring中Bean有关NullPointerException异常的原因分析》在Spring中使用@Autowired注解注入的bean不能在静态上下文中访问,否则会导致NullPointerE... 目录Spring中Bean有关NullPointerException异常的原因问题描述解决方案总结

python中的与时间相关的模块应用场景分析

《python中的与时间相关的模块应用场景分析》本文介绍了Python中与时间相关的几个重要模块:`time`、`datetime`、`calendar`、`timeit`、`pytz`和`dateu... 目录1. time 模块2. datetime 模块3. calendar 模块4. timeit

python-nmap实现python利用nmap进行扫描分析

《python-nmap实现python利用nmap进行扫描分析》Nmap是一个非常用的网络/端口扫描工具,如果想将nmap集成进你的工具里,可以使用python-nmap这个python库,它提供了... 目录前言python-nmap的基本使用PortScanner扫描PortScannerAsync异

基于Qt实现系统主题感知功能

《基于Qt实现系统主题感知功能》在现代桌面应用程序开发中,系统主题感知是一项重要的功能,它使得应用程序能够根据用户的系统主题设置(如深色模式或浅色模式)自动调整其外观,Qt作为一个跨平台的C++图形用... 目录【正文开始】一、使用效果二、系统主题感知助手类(SystemThemeHelper)三、实现细节

Oracle数据库执行计划的查看与分析技巧

《Oracle数据库执行计划的查看与分析技巧》在Oracle数据库中,执行计划能够帮助我们深入了解SQL语句在数据库内部的执行细节,进而优化查询性能、提升系统效率,执行计划是Oracle数据库优化器为... 目录一、什么是执行计划二、查看执行计划的方法(一)使用 EXPLAIN PLAN 命令(二)通过 S

性能分析之MySQL索引实战案例

文章目录 一、前言二、准备三、MySQL索引优化四、MySQL 索引知识回顾五、总结 一、前言 在上一讲性能工具之 JProfiler 简单登录案例分析实战中已经发现SQL没有建立索引问题,本文将一起从代码层去分析为什么没有建立索引? 开源ERP项目地址:https://gitee.com/jishenghua/JSH_ERP 二、准备 打开IDEA找到登录请求资源路径位置

JAVA智听未来一站式有声阅读平台听书系统小程序源码

智听未来,一站式有声阅读平台听书系统 🌟 开篇:遇见未来,从“智听”开始 在这个快节奏的时代,你是否渴望在忙碌的间隙,找到一片属于自己的宁静角落?是否梦想着能随时随地,沉浸在知识的海洋,或是故事的奇幻世界里?今天,就让我带你一起探索“智听未来”——这一站式有声阅读平台听书系统,它正悄悄改变着我们的阅读方式,让未来触手可及! 📚 第一站:海量资源,应有尽有 走进“智听