Index: xfa/fde/css/cfde_cssrulecollection.cpp |
diff --git a/xfa/fde/css/cfde_cssrulecollection.cpp b/xfa/fde/css/cfde_cssrulecollection.cpp |
new file mode 100644 |
index 0000000000000000000000000000000000000000..95f94cbff30447d5413fc6422869fbc7a209bf5b |
--- /dev/null |
+++ b/xfa/fde/css/cfde_cssrulecollection.cpp |
@@ -0,0 +1,146 @@ |
+// Copyright 2016 PDFium Authors. All rights reserved. |
+// Use of this source code is governed by a BSD-style license that can be |
+// found in the LICENSE file. |
+ |
+// Original code copyright 2014 Foxit Software Inc. http://www.foxitsoftware.com |
+ |
+#include "xfa/fde/css/fde_cssstyleselector.h" |
+ |
+#include <algorithm> |
+#include <memory> |
+ |
+#include "xfa/fde/css/fde_csscache.h" |
+#include "xfa/fde/css/fde_cssdeclaration.h" |
+#include "xfa/fde/css/fde_cssstyleselector.h" |
+#include "xfa/fde/css/fde_cssstylesheet.h" |
+#include "xfa/fde/css/fde_csssyntax.h" |
+#include "xfa/fxfa/app/xfa_textlayout.h" |
+ |
+#define FDE_CSSUNIVERSALHASH ('*') |
+ |
+void CFDE_CSSRuleCollection::Clear() { |
+ m_IDRules.RemoveAll(); |
+ m_TagRules.RemoveAll(); |
+ m_ClassRules.RemoveAll(); |
+ m_pUniversalRules = nullptr; |
+ m_pStaticStore = nullptr; |
+ m_iSelectors = 0; |
+} |
+ |
+CFDE_CSSRuleCollection::CFDE_CSSRuleCollection() |
+ : m_pStaticStore(nullptr), |
+ m_pUniversalRules(nullptr), |
+ m_pPersudoRules(nullptr), |
+ m_iSelectors(0) {} |
+ |
+CFDE_CSSRuleCollection::~CFDE_CSSRuleCollection() { |
+ Clear(); |
+} |
+ |
+void CFDE_CSSRuleCollection::AddRulesFrom(const CFDE_CSSStyleSheetArray& sheets, |
+ uint32_t dwMediaList, |
+ IFGAS_FontMgr* pFontMgr) { |
+ int32_t iSheets = sheets.GetSize(); |
+ for (int32_t i = 0; i < iSheets; ++i) { |
+ IFDE_CSSStyleSheet* pSheet = sheets.GetAt(i); |
+ if (uint32_t dwMatchMedia = pSheet->GetMediaList() & dwMediaList) { |
+ int32_t iRules = pSheet->CountRules(); |
+ for (int32_t j = 0; j < iRules; j++) { |
+ AddRulesFrom(pSheet, pSheet->GetRule(j), dwMatchMedia, pFontMgr); |
+ } |
+ } |
+ } |
+} |
+ |
+void CFDE_CSSRuleCollection::AddRulesFrom(IFDE_CSSStyleSheet* pStyleSheet, |
+ IFDE_CSSRule* pRule, |
+ uint32_t dwMediaList, |
+ IFGAS_FontMgr* pFontMgr) { |
+ switch (pRule->GetType()) { |
+ case FDE_CSSRULETYPE_Style: { |
+ IFDE_CSSStyleRule* pStyleRule = static_cast<IFDE_CSSStyleRule*>(pRule); |
+ CFDE_CSSDeclaration* pDeclaration = pStyleRule->GetDeclaration(); |
+ int32_t iSelectors = pStyleRule->CountSelectorLists(); |
+ for (int32_t i = 0; i < iSelectors; ++i) { |
+ CFDE_CSSSelector* pSelector = pStyleRule->GetSelectorList(i); |
+ if (pSelector->GetType() == FDE_CSSSELECTORTYPE_Persudo) { |
+ FDE_CSSRuleData* pData = NewRuleData(pSelector, pDeclaration); |
+ AddRuleTo(m_pPersudoRules, pData); |
+ continue; |
+ } |
+ if (pSelector->GetNameHash() != FDE_CSSUNIVERSALHASH) { |
+ AddRuleTo(m_TagRules, pSelector->GetNameHash(), pSelector, |
+ pDeclaration); |
+ continue; |
+ } |
+ CFDE_CSSSelector* pNext = pSelector->GetNextSelector(); |
+ if (!pNext) { |
+ FDE_CSSRuleData* pData = NewRuleData(pSelector, pDeclaration); |
+ AddRuleTo(m_pUniversalRules, pData); |
+ continue; |
+ } |
+ switch (pNext->GetType()) { |
+ case FDE_CSSSELECTORTYPE_ID: |
+ AddRuleTo(m_IDRules, pNext->GetNameHash(), pSelector, pDeclaration); |
+ break; |
+ case FDE_CSSSELECTORTYPE_Class: |
+ AddRuleTo(m_ClassRules, pNext->GetNameHash(), pSelector, |
+ pDeclaration); |
+ break; |
+ case FDE_CSSSELECTORTYPE_Descendant: |
+ case FDE_CSSSELECTORTYPE_Element: |
+ AddRuleTo(m_pUniversalRules, NewRuleData(pSelector, pDeclaration)); |
+ break; |
+ default: |
+ ASSERT(FALSE); |
+ break; |
+ } |
+ } |
+ } break; |
+ case FDE_CSSRULETYPE_Media: { |
+ IFDE_CSSMediaRule* pMediaRule = static_cast<IFDE_CSSMediaRule*>(pRule); |
+ if (pMediaRule->GetMediaList() & dwMediaList) { |
+ int32_t iRules = pMediaRule->CountRules(); |
+ for (int32_t i = 0; i < iRules; ++i) { |
+ AddRulesFrom(pStyleSheet, pMediaRule->GetRule(i), dwMediaList, |
+ pFontMgr); |
+ } |
+ } |
+ } break; |
+ default: |
+ break; |
+ } |
+} |
+ |
+void CFDE_CSSRuleCollection::AddRuleTo(CFX_MapPtrToPtr& map, |
+ uint32_t dwKey, |
+ CFDE_CSSSelector* pSel, |
+ CFDE_CSSDeclaration* pDecl) { |
+ void* pKey = (void*)(uintptr_t)dwKey; |
+ FDE_CSSRuleData* pData = NewRuleData(pSel, pDecl); |
+ FDE_CSSRuleData* pList = nullptr; |
+ if (!map.Lookup(pKey, (void*&)pList)) { |
+ map.SetAt(pKey, pData); |
+ } else if (AddRuleTo(pList, pData)) { |
+ map.SetAt(pKey, pList); |
+ } |
+} |
+ |
+FX_BOOL CFDE_CSSRuleCollection::AddRuleTo(FDE_CSSRuleData*& pList, |
+ FDE_CSSRuleData* pData) { |
+ if (pList) { |
+ pData->pNext = pList->pNext; |
+ pList->pNext = pData; |
+ return FALSE; |
+ } |
+ |
+ pList = pData; |
+ return TRUE; |
+} |
+ |
+FDE_CSSRuleData* CFDE_CSSRuleCollection::NewRuleData( |
+ CFDE_CSSSelector* pSel, |
+ CFDE_CSSDeclaration* pDecl) { |
+ return FXTARGET_NewWith(m_pStaticStore) |
+ FDE_CSSRuleData(pSel, pDecl, ++m_iSelectors); |
+} |