WebKit Inside: CSS 样式表的匹配时机

WebKit Inside: CSS 的解析 介绍了 CSS 样式表的解析过程,这篇文章继续介绍 CSS 的匹配时机。

无外部样式表

内部样式表和行内样式表本身就在 HTML 里面,解析 HTML 标签构建 DOM 树时内部样式表和行内样式就会被解析完毕。因此如果 HTML 里面只有内部样式表和行内样式,那么当 DOM 树构建完毕之后,就可以进行样式表的匹配了。

假设 HTML 里面的行内样式在 <div>标签,那么 CSS 匹配样式时机如下图所示:

image

如果 HTML 里面除了内部样式表或者行内样式,还有外部样式表,那么情形比较复杂。

由于引入外部样式表的 <link>标签可以位于 <head>标签中,也可以位于<body>标签中,这两种情形下,匹配时机不一样。

外部样式表位于 head

如果 HTML 里面有外部样式表和内部样式表,HTML 代码如下:

<html><head><meta charset='utf-8' /><title>EasyHTML</title><style text="text/css">/* 内部样式表 */div {background-color: red;}</style><!-- 外部样式表--><link rel="stylesheet" href="cs.css" /></head><body><div>kkk</div></body>
</html>

外部样式表 CSS 文件如下:

div {background-color: blue;font-size: 20px;
}

如果在 DOM 树构建完成之前,外部样式表就已经下载回来并且解析,那么,当 DOM 树构建完成之后,就可以直接进行样式表的匹配。

但是如果在 DOM 树构建完成之后,外部样式表还没有下载回来,那么即使内部样式表已经解析完成了,也不会进行任何样式表的匹配。调用堆栈如下图所示:

image

在函数 TreeResolver::resolveElement中,此时第一行 if里面 m_didSeePendingStyleSheet为真,因此不会进行任何样式的匹配。

由于没有进行样式匹配,无法构建渲染树,当然也不会布局和绘制,在外部样式表的下载过程中,页面是空白的。因此 CSS 的下载虽然不阻塞 DOM 树的构建,但是阻塞渲染。

变量m_didSeePendingStyleSheet在函数TreeResolver::resovle里面设置,如果位于 <head>标签里面的外部样式表还未下载成功,这个变量就是 true。设置好 m_didSeePendingStyleSheet变量,函数 TreeResolver::resove 最终会调用到TreeResolver::resolveElement里面。

TreeResolver::resolve相关代码如下所示:

std::unique_ptr<Update> TreeResolver::resolve()
{...// 1. 设置 m_didSeePendingStyleSheet 变量m_didSeePendingStylesheet = m_document.styleScope().hasPendingSheetsBeforeBody();...// 2. TreeResolver::resolveElement 函数由下面这个函数调用进去resolveComposedTree();...return WTFMove(m_update);
}

上面代码注释 1 处设置m_didSeePendingStyleSheet

代码注释 2 处,函数 TreeResolver::resolveComposedTree会调用到TreeResolver::resolveElement

当外部样式表下载完毕,仍会回调到函数TreeResolver::resove,调用堆栈如下:image

由于此时变量m_didSeePendingStyleSheet设置为false,样式表可以正常进行匹配。

image

外部样式表位于 body

把上面 HTML 里面的外部样式表挪到<body>标签,其他不变:

<html><head><meta charset='utf-8' /><title>EasyHTML</title><style text="text/css">/* 内部样式表 */div {background-color: red;}</style></head><body><!-- 外部样式表--><link rel="stylesheet" href="cs.css" /><div>kkk</div></body>
</html>

这种情形下的匹配时机会发生变化。

如果位于<body>标签的外部样式标在 DOM 树构建完成之前下载完成,那么匹配时机和上面位于<head>标签的外部样式表一样,也就是 DOM 树构建完成就进行匹配。

如果 DOM 树构建完成之后,位于<body>标签的外部样式表还未下载成功,此时由于内部样式表已经解析完成,WebKit 会对现有已解析样式表进行匹配,匹配完成之后会构建渲染树,相关代码如下:

void Document::resolveStyle(ResolveStyleType type)
{...Style::TreeResolver resolver(*this, WTFMove(m_pendingRenderTreeUpdate));// 1. 进行 CSS 样式表匹配auto styleUpdate = resolver.resolve();...if (styleUpdate) {// 2. 样式表匹配完成,这里会进行渲染树构建updateRenderTree(WTFMove(styleUpdate));frameView.styleAndRenderTreeDidChange();}...if (m_renderView->needsLayout())// 3. 渲染树构建完毕,这里会发起布局frameView.layoutContext().scheduleLayout();...
}

上面代码注释 1 处进行 CSS 样式表匹配。

代码注释 2 处现有已解析样式表匹配完毕,会进行渲染树的构建。

代码注释 3 处,如果条件允许,会进行布局计算。

但是很遗憾,如果位于<body>标签的外部样式表没有下载完成,因此不满足布局条件,代码运行不到上面代码注释 3 处,调用堆栈如下:

image

虽然有了渲染树,但是由于没有布局,也就不会进行绘制,在外部样式表下载过程中,页面同样是白色的。CSS 样式表下载依然阻塞渲染

下面看一下上图判断是否可以布局的代码,代码如下:

bool Document::shouldScheduleLayout() const
{...// 1. 因为 isVisuallyNonEmpty 方法返回了 false,导致了布局条件不满足if (view() && !view()->isVisuallyNonEmpty())return false;...return true;
}

上面代码注释 1 处由于方法LocalFrameView::isVisuallyNonEmpty返回了false,导致布局条件不满足。

方法LocalFrameView::isVisuallyNonEmpty代码如下:

bool isVisuallyNonEmpty() const { return m_contentQualifiesAsVisuallyNonEmpty; }

这个方法返回了变量m_contentQualifiesAsVisuallyNonEmpty的值,这个变量被设置为true的方法为LocalFrameView::checkAndDispatchDidReachVisuallyNonEmptyState,代码如下:

void LocalFrameView::checkAndDispatchDidReachVisuallyNonEmptyState()
{// 1. qualifiesAsVisuallyNonEmpty 回调函数auto qualifiesAsVisuallyNonEmpty = [&] {...// 2. isMoreContentExpected 回调函数auto isMoreContentExpected = [&]() {...auto& resourceLoader = documentLoader->cachedResourceLoader();// 3. 如果外部样式表已经下载成功,页面没有其他请求,这里返回 false,说明没有其他内容需要加载了if (!resourceLoader.requestCount())return false;// 4. 如果页面还有其他请求,代码运行到这里auto& resources = resourceLoader.allCachedResources();for (auto& resource : resources) {...if (resource.value->type() == CachedResource::Type::CSSStyleSheet || resource.value->type() == CachedResource::Type::FontResource)// 5. 如果正在加载的请求里面有样式表类型后者字体资源,那么这里返回 true,说明还需要等待这些资源加载return true;}return false;};// Finished parsing the main document and we still don't yet have enough content. Check if we might be getting some more.if (finishedParsingMainDocument)// 6. 调用 isMoreContentExpected 回调函数return !isMoreContentExpected();return false;};if (m_contentQualifiesAsVisuallyNonEmpty)return;// 7. 调用 qualifiesAsVisuallyNonEmpty 回调函数if (!qualifiesAsVisuallyNonEmpty())return;// 8. 这里设置 m_contentQualifiesAsVisuallyNonEmpty 为 truem_contentQualifiesAsVisuallyNonEmpty = true;...
}

上面代码注释 1 处定义了qualifiesAsVisuallyNonEmpty回调函数。

代码注释 2 定义了isMoreContentExpected回调函数。

代码注释 7 处调用了回调函数qualifiesAsVisuallyNonEmpty

qualifiesAsVisuallyNonEmpty回调函数里面,调用了回调函数isMoreContentExpected,如代码注释 6 所示。

回调函数isMoreContentExpected里面会判断当前是否还有其他请求,如果代码注释 3 所示。如果没有其他请求了,isMoreContentExpected 函数返回 false,表明没有其他内容要加载了。因此,此时代码会运行到代码注释 8 处,将变量m_contentQualifiesAsVisuallyNonEmpty设置为true

如果页面还有其他资源的请求,比如外部样式表还在请求,那么回调函数isMoreContentExpected会运行到代码注释 5 处。这里会判断请求资源类型是否是样式表或者字体资源,如果是这两种资源之一,这里返回 true。这样,代码会运行到注释 7 处,直接返回而不设置变量m_contentQualifiesAsVisuallyNonEmpty

因此,如果位于<body>标签的外部样式表还在下载,那么就会在上面代码注释 7 返回,所以不会进行布局。

如果外部样式表下载成功并解析之后,会调用Document::resolveStyle方法,这个方法会进行样式表的匹配,渲染树的构建,布局的调用,代码如下:

void Document::resolveStyle(ResolveStyleType type)
{...Style::TreeResolver resolver(*this, WTFMove(m_pendingRenderTreeUpdate));// 1. 样式表匹配auto styleUpdate = resolver.resolve();...if (styleUpdate) {// 2. 构建渲染树updateRenderTree(WTFMove(styleUpdate));// 3. 设置 m_contentQualifiesAsVisuallyNonEmpty = true 的方法在这里调用frameView.styleAndRenderTreeDidChange();}...if (m_renderView->needsLayout())// 4. 调用布局方法frameView.layoutContext().scheduleLayout();...
}

上面代码注释 1 处进行样式表匹配。

代码注释 2 进行渲染树构建。

代码注释 3 这个方法内部会调用LocalFrameView::checkAndDispatchDidReachVisuallyNonEmptyState方法设置变量m_contentQualifiesAsVisuallyNonEmpty。由于外部样式表已经下载成功,此时变量m_contentQualifiesAsVisuallyNonEmpty就会被设置成true

由于上面的设置,后续代码注释 4 处的布局方法调用就可以成功了。

这种情形下匹配时机如下图所示:image

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/97959.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【Qt】顶层窗口和普通窗口区别以及用法

区别 在Qt项目开发中&#xff0c;经常会用到窗体控件用于显示及数据操作和其他交互等。 但&#xff0c;窗体分为顶层窗口&#xff08;Top-level Window&#xff09;和普通窗口&#xff08;Regular Window&#xff09;。 他们之间是有区别的&#xff0c;包括在项目实际中的用法…

【力扣面试题】URL化

&#x1f451;专栏内容&#xff1a;力扣刷题⛪个人主页&#xff1a;子夜的星的主页&#x1f495;座右铭&#xff1a;前路未远&#xff0c;步履不停 目录 一、题目描述二、题目分析1、使用String内部方法2、使用StringBuilder 一、题目描述 题目链接&#xff1a;URL化 编写一种…

邮件群发工具哪个好

邮件群发是一种通过电子邮件向多个收件人发送邮件的方式。同时&#xff0c;邮件群发也是一种低成本、高回报的营销手段。因此邮件群发被广泛应用于各种营销活动中&#xff0c;例如活动邀请、新品上线、产品促销等等。而群发邮件最有效的方式就是借助邮件群发工具&#xff0c;而…

文本自动输入/删除的加载动画效果

效果展示 CSS 知识点 绕矩形四周跑的光柱动画实现animation 属性的 steps 属性值运用 页面基础结构实现 <div class"loader"><!-- span 标签是围绕矩形四周的光柱 --><span></span><span></span><span></span>&l…

Scratch3.0下载

通俗易懂&#xff0c;直接上链接 链接&#xff1a;https://pan.baidu.com/s/1n-QFEQWT8im8BHQu1wIjtg?pwd1016 提取码&#xff1a;1016

zookeeper选举机制

全新集群选举 zookeeper 全新集群选举机制网上资料很多说法很模糊&#xff0c;仔细思考了一下&#xff0c;应该是这样 得到票数最多的机器>机器总数半数 具体启动过程中的哪个节点成为 leader 与 zoo.cfg 中配置的节点数有关&#xff0c;下面以3个举例 选举过程如下 server…

tailscale自建headscale和derp中继

tailscale derp中继服务简介 tailscale是一个基于WireGuard的零配置软件&#xff0c;它可以轻松地在多台设备之间建立点对点加密连接。 derp服务器是tailscale网络的重要组成部分。它作为tailscale客户端之间的中继,帮助客户端找到并连接到其他客户端设备。 但Tailscale 官方…

微信小程序wxs标签 在wxml文件中编写JavaScript逻辑

PC端开发 可以在界面中编写JavaScript脚本 vue/react这些框架更是形成了一种常态 因为模板引擎和jsx语法本身就都是在js中的 我们小程序中其实也有类似的奇妙写法 不过先声明 这东西不是很强大 我们可以先写一个案例代码 wxml代码参考 <view><wxs module"wordSt…

MySQL命令行中文乱码问题

MySQL命令行中文乱码问题&#xff1a; 命令行界面默认字符集是gbk&#xff0c;若字符集不匹配会中文乱码或无法插入中文。 解决办法&#xff1a;执行set names gbk; 验证&#xff1a; 执行命令show variables like ‘char%’;查看默认字符集。 创建数据库设置字符集utf8&…

自动驾驶:未来的道路上的挑战与机遇

自动驾驶&#xff1a;未来的道路上的挑战与机遇 文章目录 引言安全与道路事故的减少交通拥堵的缓解城市规划的变革技术和法律挑战结语 2023星火培训【专项营】Apollo开发者社区布道师倾力打造&#xff0c;包含PnC、新感知等的全新专项课程上线了。理论与实践相结合&#xff0c;…

1.3.2有理数减法(第一课时)作业设计

【学习目标】 1&#xff0e;理解有理数减法法则&#xff0c;能熟练地进行有理数的减法运算&#xff0e; 2&#xff0e;感受有理数减法与加法对立统一的辨证思想&#xff0c;体会转化的思想方法&#xff0e;

请问python如何处理url带有“?”参数的接口?

在Python中处理带有"?"参数的URL接口&#xff0c;可以使用urllib.parse库中的urlencode()函数来进行编码。以下是一些示例代码 from urllib.parse import urlencodeparams {name: John, age: 25} url http://example.com? urlencode(params) print(url) 这个代…

泛微OA e-office平台uploadify.php任意文件上传漏洞

泛微OA e-office平台uploadify.php任意文件上传漏洞复现 0x01 前言 免责声明&#xff1a;请勿利用文章内的相关技术从事非法测试&#xff0c;由于传播、利用此文所提供的信息或者工具而造成的任何直接或者间接的后果及损失&#xff0c;均由使用者本人负责&#xff0c;所产生的…

NAT模式和桥接模式的区别

NAT模式和桥接模式的区别 NAT模式和桥接模式都是虚拟机网络配置的两种方式&#xff0c;主要区别在于虚拟机与外部网络交互的方式不同。 NAT&#xff08;Network Address Translation&#xff0c;网络地址转换&#xff09;模式&#xff1a;在这种模式下&#xff0c;虚拟机和宿主…

爬虫编程语言

文章目录 基本数据类型bytes类型python数据类型转换 python运算符python数字数学函数随机数函数三角函数数字常量 python字符串python访问字符串中的值python字符串更新python转义字符python字符串运算符python字符串格式化f-stringUnicode字符串python的字符串内建函数 python…

【轻松玩转MacOS】外部设备篇

引言 在开始之前&#xff0c;我们先来了解一下为什么要连接外部设备。想象一下&#xff0c;你正在享受MacOS带来的便捷和高效&#xff0c;突然需要打印一份文件&#xff0c;但你发现打印机无法连接&#xff1b;或者你需要将手机投屏到电脑上&#xff0c;却不知道该如何操作。这…

Docker搭建MySQL8.0主从复制(一主一从)

0. 配置说明 宿主机使用的版本为19045的win10专业版&#xff0c;MySQL使用的是8.0&#xff0c;Docker容器使用Linux。 1. 安装Docker Desktop 略 修改Docker默认安装路径 安装包自己就提供了修改安装路径的功能&#xff0c;CMD中运行&#xff1a; “Docker Desktop Installe…

【网络安全-信息收集】网络安全之信息收集和信息收集工具讲解(提供工具)

工具下载百度网盘链接(包含所有用到的工具&#xff09;&#xff1a; 百度网盘 请输入提取码百度网盘为您提供文件的网络备份、同步和分享服务。空间大、速度快、安全稳固&#xff0c;支持教育网加速&#xff0c;支持手机端。注册使用百度网盘即可享受免费存储空间https://pan.…

HDLbits: ece241 2014 q4

module top_module (input clk,input x,output z ); reg [2:0] Q;always(posedge clk)beginQ[0] < Q[0] ^ x;Q[1] < (~Q[1]) & x;Q[2] < (~Q[2]) | x;z < ~(| Q[2:0]); //错误&#xff01;&#xff01;&#xff01;&#xff01;endendmodule 正确答案&#xf…

10.07hw

int main() {string str;cout << "输入字符串:";getline(cin,str);int size str.size();int big 0;int small 0;int num 0;int space 0;int other 0;for(int i0;i<size;i){if(str[i]<Z&&str[i]>A){big;}else if(str[i]<z&&st…