在浏览器内核(排版引擎)CSS规则树和HTML的DOM树合成渲染树的时候,会涉及到渲染树的位置属性的问题,因为其位置属性将通过CSS选择器链的优先级来决定,而渲染树的某个结点可能会同时满足多个选择器链,这时候就要通过选择器的优先级来完成属性的赋值。
在这个地方,我仅仅处理了几个简单的选择器情况:{(.class) (#id) (element) (#id,.class,elememt) (#id>.class) (#id element) (#id.class)}。
这时候,将渲染结点同时满足的几个选择器链通过其优先级加权算值,从小到大依次覆盖渲染结点;而如何确定此渲染结点是否满足某个选择器链呢?
我的实现过程大致如下:
首先给定选择器的结构体如下:
struct CSSParserSelector { enum TYPE {ID,CLASS,ELEMENT}; enum RELATION {NONE,DESCENDANT,CHILD,AND}; TYPE m_Type; std::string m_Name; RELATION m_Relation; struct CSSParserSelector *ptr; CSSParserSelector():m_Relation(NONE),ptr(NULL) {} };
从本渲染结点开始,判断此结点是否与选择器链表的当前选择器相匹配。如果匹配,判断此选择器与下一个选择器的关系:如果为NONE,表示本选择器是选择器链的最后一个,返回成功;如果关系为AND (比如:#id.class),选择下一个选择器与本渲染结点继续比较;如果关系为CHILD,表示本选择器是下一个选择器的子结点,返回下一个选择器与下一个渲染结点的匹配结果;否则,关系为DESCENDANT,选择器和渲染结点各指向下一个结点,然后将渲染结点继续回溯,直到第一个满足回溯后的选择器的结点,此时将继续判断回溯后的选择器和回溯后的渲染结点是否匹配。这个过程大致如下:
/** *选择器是否与渲染结点匹配 */ bool RenderTree::IsSelectorsMatchRenderTree(CSSParserSelector *selector,RenderNode *renderNode) { if(IsSelectorMatchRenderNode(selector,renderNode)) { if(selector->m_Relation == CSSParserSelector::NONE) { return true; } else if(selector->m_Relation == CSSParserSelector::AND) { return IsSelectorsMatchRenderTree(selector->ptr,renderNode); } else if(selector->m_Relation == CSSParserSelector::CHILD) { return IsSelectorsMatchRenderTree(selector->ptr,renderNode->parentNode); } else { return IsMatchDescendantRelation(selector->ptr,renderNode->parentNode); } } else { return false; } } /** * 判断与祖父关系是否匹配 */ bool RenderTree::IsMatchDescendantRelation(CSSParserSelector *selector,RenderNode *renderNode) { while(renderNode->getTypeOfRenderNode() != "html" && (!IsSelectorMatchRenderNode(selector,renderNode))) { renderNode = renderNode->parentNode; } if(renderNode->getTypeOfRenderNode() != "html") { return IsSelectorsMatchRenderTree(selector,renderNode); } else { return false; } }最终便得到了匹配结果。而后通过公式便可以计算选择器链表的优先级。
CSS规则树和HTML的DOM树合成渲染树时渲染结点与选择器链的匹配
原文地址:http://blog.csdn.net/u012637838/article/details/41900339