| // Copyright 2014 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_cssstylesheet.h" |
| |
| #include <memory> |
| |
| #include "xfa/fde/css/fde_cssdatatable.h" |
| #include "xfa/fde/css/fde_csssyntax.h" |
| #include "xfa/fgas/crt/fgas_codepage.h" |
| |
| IFDE_CSSStyleSheet* IFDE_CSSStyleSheet::LoadHTMLStandardStyleSheet() { |
| static const FX_WCHAR s_pStyle[] = |
| L"html,address,blockquote,body,dd,div,dl,dt,fieldset,form,frame,frameset," |
| L"h1,h2,h3,h4,h5,h6,noframes,ol,p,ul,center,dir,hr,menu,pre{display:" |
| L"block}" |
| L"li{display:list-item}head{display:none}table{display:table}tr{display:" |
| L"table-row}thead{display:table-header-group}tbody{display:table-row-" |
| L"group}tfoot{display:table-footer-group}" |
| L"col{display:table-column}colgroup{display:table-column-group}td,th{" |
| L"display:table-cell}caption{display:table-caption}th{font-weight:bolder;" |
| L"text-align:center}caption{text-align:center}" |
| L"body{margin:0}h1{font-size:2em;margin:.67em " |
| L"0}h2{font-size:1.5em;margin:.75em 0}h3{font-size:1.17em;margin:.83em " |
| L"0}h4,p,blockquote,ul,fieldset,form,ol,dl,dir,menu{margin:1.12em 0}" |
| L"h5{font-size:.83em;margin:1.5em 0}h6{font-size:.75em;margin:1.67em " |
| L"0}h1,h2,h3,h4,h5,h6,b,strong{font-weight:bolder}blockquote{margin-left:" |
| L"40px;margin-right:40px}i,cite,em,var,address{font-style:italic}" |
| L"pre,tt,code,kbd,samp{font-family:monospace}pre{white-space:pre}button," |
| L"textarea,input,select{display:inline-block}big{font-size:1.17em}small," |
| L"sub,sup{font-size:.83em}sub{vertical-align:sub}" |
| L"sup{vertical-align:super}table{border-spacing:2px}thead,tbody,tfoot{" |
| L"vertical-align:middle}td,th,tr{vertical-align:inherit}s,strike,del{" |
| L"text-decoration:line-through}hr{border:1px inset silver}" |
| L"ol,ul,dir,menu,dd{margin-left:40px}ol{list-style-type:decimal}ol ul,ul " |
| L"ol,ul ul,ol " |
| L"ol{margin-top:0;margin-bottom:0}u,ins{text-decoration:underline}center{" |
| L"text-align:center}" |
| L"ruby{display:ruby}rt{display:ruby-text;font-size:.5em}rb{display:ruby-" |
| L"base}rbc{display:ruby-base-group}rtc{display:ruby-text-group}" |
| L"q:before{content:open-quote}q:after{content:close-quote}" |
| L"rp{display:none}"; |
| return IFDE_CSSStyleSheet::LoadFromBuffer( |
| CFX_WideString(), s_pStyle, FXSYS_wcslen(s_pStyle), FX_CODEPAGE_UTF8); |
| } |
| |
| IFDE_CSSStyleSheet* IFDE_CSSStyleSheet::LoadFromStream( |
| const CFX_WideString& szUrl, |
| IFX_Stream* pStream, |
| uint16_t wCodePage, |
| uint32_t dwMediaList) { |
| CFDE_CSSStyleSheet* pStyleSheet = new CFDE_CSSStyleSheet(dwMediaList); |
| if (!pStyleSheet->LoadFromStream(szUrl, pStream, wCodePage)) { |
| pStyleSheet->Release(); |
| pStyleSheet = nullptr; |
| } |
| return pStyleSheet; |
| } |
| |
| IFDE_CSSStyleSheet* IFDE_CSSStyleSheet::LoadFromBuffer( |
| const CFX_WideString& szUrl, |
| const FX_WCHAR* pBuffer, |
| int32_t iBufSize, |
| uint16_t wCodePage, |
| uint32_t dwMediaList) { |
| CFDE_CSSStyleSheet* pStyleSheet = new CFDE_CSSStyleSheet(dwMediaList); |
| if (!pStyleSheet->LoadFromBuffer(szUrl, pBuffer, iBufSize, wCodePage)) { |
| pStyleSheet->Release(); |
| pStyleSheet = nullptr; |
| } |
| return pStyleSheet; |
| } |
| |
| CFDE_CSSStyleSheet::CFDE_CSSStyleSheet(uint32_t dwMediaList) |
| : m_wCodePage(FX_CODEPAGE_UTF8), |
| m_wRefCount(1), |
| m_dwMediaList(dwMediaList), |
| m_pAllocator(nullptr) { |
| ASSERT(m_dwMediaList > 0); |
| } |
| |
| CFDE_CSSStyleSheet::~CFDE_CSSStyleSheet() { |
| Reset(); |
| } |
| |
| void CFDE_CSSStyleSheet::Reset() { |
| for (int32_t i = m_RuleArray.GetSize() - 1; i >= 0; --i) { |
| IFDE_CSSRule* pRule = m_RuleArray.GetAt(i); |
| switch (pRule->GetType()) { |
| case FDE_CSSRULETYPE_Style: |
| static_cast<CFDE_CSSStyleRule*>(pRule)->~CFDE_CSSStyleRule(); |
| break; |
| case FDE_CSSRULETYPE_Media: |
| static_cast<CFDE_CSSMediaRule*>(pRule)->~CFDE_CSSMediaRule(); |
| break; |
| case FDE_CSSRULETYPE_FontFace: |
| static_cast<CFDE_CSSFontFaceRule*>(pRule)->~CFDE_CSSFontFaceRule(); |
| break; |
| default: |
| ASSERT(FALSE); |
| break; |
| } |
| } |
| m_RuleArray.RemoveAll(); |
| m_Selectors.RemoveAll(); |
| m_StringCache.clear(); |
| delete m_pAllocator; |
| m_pAllocator = nullptr; |
| } |
| |
| uint32_t CFDE_CSSStyleSheet::Retain() { |
| return ++m_wRefCount; |
| } |
| |
| uint32_t CFDE_CSSStyleSheet::Release() { |
| uint32_t dwRefCount = --m_wRefCount; |
| if (dwRefCount == 0) { |
| delete this; |
| } |
| return dwRefCount; |
| } |
| |
| FX_BOOL CFDE_CSSStyleSheet::GetUrl(CFX_WideString& szUrl) { |
| szUrl = m_szUrl; |
| return szUrl.GetLength() > 0; |
| } |
| |
| uint32_t CFDE_CSSStyleSheet::GetMediaList() const { |
| return m_dwMediaList; |
| } |
| |
| uint16_t CFDE_CSSStyleSheet::GetCodePage() const { |
| return m_wCodePage; |
| } |
| |
| int32_t CFDE_CSSStyleSheet::CountRules() const { |
| return m_RuleArray.GetSize(); |
| } |
| |
| IFDE_CSSRule* CFDE_CSSStyleSheet::GetRule(int32_t index) { |
| return m_RuleArray.GetAt(index); |
| } |
| |
| FX_BOOL CFDE_CSSStyleSheet::LoadFromStream(const CFX_WideString& szUrl, |
| IFX_Stream* pStream, |
| uint16_t wCodePage) { |
| std::unique_ptr<CFDE_CSSSyntaxParser> pSyntax(new CFDE_CSSSyntaxParser); |
| if (pStream->GetCodePage() != wCodePage) |
| pStream->SetCodePage(wCodePage); |
| |
| FX_BOOL bRet = pSyntax->Init(pStream, 4096) && LoadFromSyntax(pSyntax.get()); |
| m_wCodePage = wCodePage; |
| m_szUrl = szUrl; |
| return bRet; |
| } |
| |
| FX_BOOL CFDE_CSSStyleSheet::LoadFromBuffer(const CFX_WideString& szUrl, |
| const FX_WCHAR* pBuffer, |
| int32_t iBufSize, |
| uint16_t wCodePage) { |
| ASSERT(pBuffer && iBufSize > 0); |
| std::unique_ptr<CFDE_CSSSyntaxParser> pSyntax(new CFDE_CSSSyntaxParser); |
| FX_BOOL bRet = |
| pSyntax->Init(pBuffer, iBufSize) && LoadFromSyntax(pSyntax.get()); |
| m_wCodePage = wCodePage; |
| m_szUrl = szUrl; |
| return bRet; |
| } |
| |
| FX_BOOL CFDE_CSSStyleSheet::LoadFromSyntax(CFDE_CSSSyntaxParser* pSyntax) { |
| Reset(); |
| m_pAllocator = IFX_MemoryAllocator::Create(FX_ALLOCTYPE_Static, 1024, 0); |
| FDE_CSSSYNTAXSTATUS eStatus; |
| do { |
| switch (eStatus = pSyntax->DoSyntaxParse()) { |
| case FDE_CSSSYNTAXSTATUS_StyleRule: |
| eStatus = LoadStyleRule(pSyntax, m_RuleArray); |
| break; |
| case FDE_CSSSYNTAXSTATUS_MediaRule: |
| eStatus = LoadMediaRule(pSyntax); |
| break; |
| case FDE_CSSSYNTAXSTATUS_FontFaceRule: |
| eStatus = LoadFontFaceRule(pSyntax, m_RuleArray); |
| break; |
| case FDE_CSSSYNTAXSTATUS_ImportRule: |
| eStatus = LoadImportRule(pSyntax); |
| break; |
| case FDE_CSSSYNTAXSTATUS_PageRule: |
| eStatus = LoadPageRule(pSyntax); |
| break; |
| default: |
| break; |
| } |
| } while (eStatus >= FDE_CSSSYNTAXSTATUS_None); |
| m_Selectors.RemoveAll(); |
| m_StringCache.clear(); |
| return eStatus != FDE_CSSSYNTAXSTATUS_Error; |
| } |
| |
| FDE_CSSSYNTAXSTATUS CFDE_CSSStyleSheet::LoadMediaRule( |
| CFDE_CSSSyntaxParser* pSyntax) { |
| uint32_t dwMediaList = 0; |
| CFDE_CSSMediaRule* pMediaRule = nullptr; |
| for (;;) { |
| switch (pSyntax->DoSyntaxParse()) { |
| case FDE_CSSSYNTAXSTATUS_MediaType: { |
| int32_t iLen; |
| const FX_WCHAR* psz = pSyntax->GetCurrentString(iLen); |
| FDE_LPCCSSMEDIATYPETABLE pMediaType = |
| FDE_GetCSSMediaTypeByName(CFX_WideStringC(psz, iLen)); |
| if (pMediaType) |
| dwMediaList |= pMediaType->wValue; |
| } break; |
| case FDE_CSSSYNTAXSTATUS_StyleRule: |
| if (pMediaRule) { |
| FDE_CSSSYNTAXSTATUS eStatus = |
| LoadStyleRule(pSyntax, pMediaRule->GetArray()); |
| if (eStatus < FDE_CSSSYNTAXSTATUS_None) { |
| return eStatus; |
| } |
| } else { |
| SkipRuleSet(pSyntax); |
| } |
| break; |
| case FDE_CSSSYNTAXSTATUS_DeclOpen: |
| if ((dwMediaList & m_dwMediaList) > 0 && !pMediaRule) { |
| pMediaRule = |
| FXTARGET_NewWith(m_pAllocator) CFDE_CSSMediaRule(dwMediaList); |
| m_RuleArray.Add(pMediaRule); |
| } |
| break; |
| case FDE_CSSSYNTAXSTATUS_DeclClose: |
| return FDE_CSSSYNTAXSTATUS_None; |
| FDE_CSSSWITCHDEFAULTS(); |
| } |
| } |
| } |
| |
| FDE_CSSSYNTAXSTATUS CFDE_CSSStyleSheet::LoadStyleRule( |
| CFDE_CSSSyntaxParser* pSyntax, |
| CFDE_CSSRuleArray& ruleArray) { |
| m_Selectors.RemoveAt(0, m_Selectors.GetSize()); |
| CFDE_CSSStyleRule* pStyleRule = nullptr; |
| const FX_WCHAR* pszValue = nullptr; |
| int32_t iValueLen = 0; |
| FDE_CSSPROPERTYARGS propertyArgs; |
| propertyArgs.pStaticStore = m_pAllocator; |
| propertyArgs.pStringCache = &m_StringCache; |
| propertyArgs.pProperty = nullptr; |
| CFX_WideString wsName; |
| for (;;) { |
| switch (pSyntax->DoSyntaxParse()) { |
| case FDE_CSSSYNTAXSTATUS_Selector: { |
| pszValue = pSyntax->GetCurrentString(iValueLen); |
| CFDE_CSSSelector* pSelector = |
| CFDE_CSSSelector::FromString(m_pAllocator, pszValue, iValueLen); |
| if (pSelector) |
| m_Selectors.Add(pSelector); |
| } break; |
| case FDE_CSSSYNTAXSTATUS_PropertyName: |
| pszValue = pSyntax->GetCurrentString(iValueLen); |
| propertyArgs.pProperty = |
| FDE_GetCSSPropertyByName(CFX_WideStringC(pszValue, iValueLen)); |
| if (!propertyArgs.pProperty) |
| wsName = CFX_WideStringC(pszValue, iValueLen); |
| break; |
| case FDE_CSSSYNTAXSTATUS_PropertyValue: |
| if (propertyArgs.pProperty) { |
| pszValue = pSyntax->GetCurrentString(iValueLen); |
| if (iValueLen > 0) { |
| pStyleRule->GetDeclImp().AddProperty(&propertyArgs, pszValue, |
| iValueLen); |
| } |
| } else if (iValueLen > 0) { |
| pszValue = pSyntax->GetCurrentString(iValueLen); |
| if (iValueLen > 0) { |
| pStyleRule->GetDeclImp().AddProperty(&propertyArgs, wsName.c_str(), |
| wsName.GetLength(), pszValue, |
| iValueLen); |
| } |
| } |
| break; |
| case FDE_CSSSYNTAXSTATUS_DeclOpen: |
| if (!pStyleRule && m_Selectors.GetSize() > 0) { |
| pStyleRule = FXTARGET_NewWith(m_pAllocator) CFDE_CSSStyleRule; |
| pStyleRule->SetSelector(m_pAllocator, m_Selectors); |
| ruleArray.Add(pStyleRule); |
| } else { |
| SkipRuleSet(pSyntax); |
| return FDE_CSSSYNTAXSTATUS_None; |
| } |
| break; |
| case FDE_CSSSYNTAXSTATUS_DeclClose: |
| if (pStyleRule && !pStyleRule->GetDeclImp().GetStartPosition()) { |
| pStyleRule->~CFDE_CSSStyleRule(); |
| ruleArray.RemoveLast(1); |
| } |
| return FDE_CSSSYNTAXSTATUS_None; |
| FDE_CSSSWITCHDEFAULTS(); |
| } |
| } |
| } |
| |
| FDE_CSSSYNTAXSTATUS CFDE_CSSStyleSheet::LoadFontFaceRule( |
| CFDE_CSSSyntaxParser* pSyntax, |
| CFDE_CSSRuleArray& ruleArray) { |
| CFDE_CSSFontFaceRule* pFontFaceRule = nullptr; |
| const FX_WCHAR* pszValue = nullptr; |
| int32_t iValueLen = 0; |
| FDE_CSSPROPERTYARGS propertyArgs; |
| propertyArgs.pStaticStore = m_pAllocator; |
| propertyArgs.pStringCache = &m_StringCache; |
| propertyArgs.pProperty = nullptr; |
| for (;;) { |
| switch (pSyntax->DoSyntaxParse()) { |
| case FDE_CSSSYNTAXSTATUS_PropertyName: |
| pszValue = pSyntax->GetCurrentString(iValueLen); |
| propertyArgs.pProperty = |
| FDE_GetCSSPropertyByName(CFX_WideStringC(pszValue, iValueLen)); |
| break; |
| case FDE_CSSSYNTAXSTATUS_PropertyValue: |
| if (propertyArgs.pProperty) { |
| pszValue = pSyntax->GetCurrentString(iValueLen); |
| if (iValueLen > 0) { |
| pFontFaceRule->GetDeclImp().AddProperty(&propertyArgs, pszValue, |
| iValueLen); |
| } |
| } |
| break; |
| case FDE_CSSSYNTAXSTATUS_DeclOpen: |
| if (!pFontFaceRule) { |
| pFontFaceRule = FXTARGET_NewWith(m_pAllocator) CFDE_CSSFontFaceRule; |
| ruleArray.Add(pFontFaceRule); |
| } |
| break; |
| case FDE_CSSSYNTAXSTATUS_DeclClose: |
| return FDE_CSSSYNTAXSTATUS_None; |
| FDE_CSSSWITCHDEFAULTS(); |
| } |
| } |
| } |
| |
| FDE_CSSSYNTAXSTATUS CFDE_CSSStyleSheet::LoadImportRule( |
| CFDE_CSSSyntaxParser* pSyntax) { |
| for (;;) { |
| switch (pSyntax->DoSyntaxParse()) { |
| case FDE_CSSSYNTAXSTATUS_ImportClose: |
| return FDE_CSSSYNTAXSTATUS_None; |
| case FDE_CSSSYNTAXSTATUS_URI: |
| break; |
| FDE_CSSSWITCHDEFAULTS(); |
| } |
| } |
| } |
| |
| FDE_CSSSYNTAXSTATUS CFDE_CSSStyleSheet::LoadPageRule( |
| CFDE_CSSSyntaxParser* pSyntax) { |
| return SkipRuleSet(pSyntax); |
| } |
| |
| FDE_CSSSYNTAXSTATUS CFDE_CSSStyleSheet::SkipRuleSet( |
| CFDE_CSSSyntaxParser* pSyntax) { |
| for (;;) { |
| switch (pSyntax->DoSyntaxParse()) { |
| case FDE_CSSSYNTAXSTATUS_Selector: |
| case FDE_CSSSYNTAXSTATUS_DeclOpen: |
| case FDE_CSSSYNTAXSTATUS_PropertyName: |
| case FDE_CSSSYNTAXSTATUS_PropertyValue: |
| break; |
| case FDE_CSSSYNTAXSTATUS_DeclClose: |
| return FDE_CSSSYNTAXSTATUS_None; |
| FDE_CSSSWITCHDEFAULTS(); |
| } |
| } |
| } |
| |
| CFDE_CSSStyleRule::CFDE_CSSStyleRule() |
| : m_ppSelector(nullptr), m_iSelectors(0) {} |
| |
| int32_t CFDE_CSSStyleRule::CountSelectorLists() const { |
| return m_iSelectors; |
| } |
| |
| CFDE_CSSSelector* CFDE_CSSStyleRule::GetSelectorList(int32_t index) const { |
| return m_ppSelector[index]; |
| } |
| |
| CFDE_CSSDeclaration* CFDE_CSSStyleRule::GetDeclaration() { |
| return &m_Declaration; |
| } |
| |
| void CFDE_CSSStyleRule::SetSelector(IFX_MemoryAllocator* pStaticStore, |
| const CFDE_CSSSelectorArray& list) { |
| ASSERT(!m_ppSelector); |
| m_iSelectors = list.GetSize(); |
| m_ppSelector = static_cast<CFDE_CSSSelector**>( |
| pStaticStore->Alloc(m_iSelectors * sizeof(CFDE_CSSSelector*))); |
| for (int32_t i = 0; i < m_iSelectors; ++i) { |
| m_ppSelector[i] = list.GetAt(i); |
| } |
| } |
| |
| CFDE_CSSMediaRule::CFDE_CSSMediaRule(uint32_t dwMediaList) |
| : m_dwMediaList(dwMediaList) {} |
| |
| CFDE_CSSMediaRule::~CFDE_CSSMediaRule() { |
| for (int32_t i = m_RuleArray.GetSize() - 1; i >= 0; --i) { |
| IFDE_CSSRule* pRule = m_RuleArray.GetAt(i); |
| switch (pRule->GetType()) { |
| case FDE_CSSRULETYPE_Style: |
| ((CFDE_CSSStyleRule*)pRule)->~CFDE_CSSStyleRule(); |
| break; |
| default: |
| ASSERT(FALSE); |
| break; |
| } |
| } |
| } |
| |
| uint32_t CFDE_CSSMediaRule::GetMediaList() const { |
| return m_dwMediaList; |
| } |
| |
| int32_t CFDE_CSSMediaRule::CountRules() const { |
| return m_RuleArray.GetSize(); |
| } |
| |
| IFDE_CSSRule* CFDE_CSSMediaRule::GetRule(int32_t index) { |
| return m_RuleArray.GetAt(index); |
| } |
| |
| FX_BOOL FDE_IsCSSChar(FX_WCHAR wch) { |
| return (wch >= 'a' && wch <= 'z') || (wch >= 'A' && wch <= 'Z'); |
| } |
| |
| int32_t FDE_GetCSSPersudoLen(const FX_WCHAR* psz, const FX_WCHAR* pEnd) { |
| ASSERT(*psz == ':'); |
| const FX_WCHAR* pStart = psz; |
| while (psz < pEnd) { |
| FX_WCHAR wch = *psz; |
| if (FDE_IsCSSChar(wch) || wch == ':') { |
| ++psz; |
| } else { |
| break; |
| } |
| } |
| return psz - pStart; |
| } |
| |
| int32_t FDE_GetCSSNameLen(const FX_WCHAR* psz, const FX_WCHAR* pEnd) { |
| const FX_WCHAR* pStart = psz; |
| while (psz < pEnd) { |
| FX_WCHAR wch = *psz; |
| if (FDE_IsCSSChar(wch) || (wch >= '0' && wch <= '9') || wch == '_' || |
| wch == '-') { |
| ++psz; |
| } else { |
| break; |
| } |
| } |
| return psz - pStart; |
| } |
| |
| CFDE_CSSSelector::CFDE_CSSSelector(FDE_CSSSELECTORTYPE eType, |
| const FX_WCHAR* psz, |
| int32_t iLen, |
| bool bIgnoreCase) |
| : m_eType(eType), |
| m_dwHash(FX_HashCode_GetW(CFX_WideStringC(psz, iLen), bIgnoreCase)), |
| m_pNext(nullptr) {} |
| |
| FDE_CSSSELECTORTYPE CFDE_CSSSelector::GetType() const { |
| return m_eType; |
| } |
| |
| uint32_t CFDE_CSSSelector::GetNameHash() const { |
| return m_dwHash; |
| } |
| |
| CFDE_CSSSelector* CFDE_CSSSelector::GetNextSelector() const { |
| return m_pNext; |
| } |
| |
| CFDE_CSSSelector* CFDE_CSSSelector::FromString( |
| IFX_MemoryAllocator* pStaticStore, |
| const FX_WCHAR* psz, |
| int32_t iLen) { |
| ASSERT(pStaticStore && psz && iLen > 0); |
| |
| const FX_WCHAR* pStart = psz; |
| const FX_WCHAR* pEnd = psz + iLen; |
| for (; psz < pEnd; ++psz) { |
| switch (*psz) { |
| case '>': |
| case '[': |
| case '+': |
| return nullptr; |
| } |
| } |
| CFDE_CSSSelector* pFirst = nullptr; |
| CFDE_CSSSelector* pLast = nullptr; |
| CFDE_CSSSelector* pPersudoFirst = nullptr; |
| CFDE_CSSSelector* pPersudoLast = nullptr; |
| for (psz = pStart; psz < pEnd;) { |
| FX_WCHAR wch = *psz; |
| if (wch == '.' || wch == '#') { |
| if (psz == pStart || psz[-1] == ' ') { |
| CFDE_CSSSelector* p = FXTARGET_NewWith(pStaticStore) |
| CFDE_CSSSelector(FDE_CSSSELECTORTYPE_Element, L"*", 1, true); |
| if (!p) |
| return nullptr; |
| |
| if (pFirst) { |
| pFirst->SetType(FDE_CSSSELECTORTYPE_Descendant); |
| p->SetNext(pFirst); |
| } |
| pFirst = pLast = p; |
| } |
| ASSERT(pLast); |
| int32_t iNameLen = FDE_GetCSSNameLen(++psz, pEnd); |
| if (iNameLen == 0) { |
| return nullptr; |
| } |
| FDE_CSSSELECTORTYPE eType = |
| wch == '.' ? FDE_CSSSELECTORTYPE_Class : FDE_CSSSELECTORTYPE_ID; |
| CFDE_CSSSelector* p = FXTARGET_NewWith(pStaticStore) |
| CFDE_CSSSelector(eType, psz, iNameLen, false); |
| if (!p) |
| return nullptr; |
| |
| p->SetNext(pLast->GetNextSelector()); |
| pLast->SetNext(p); |
| pLast = p; |
| psz += iNameLen; |
| } else if (FDE_IsCSSChar(wch) || wch == '*') { |
| int32_t iNameLen = wch == '*' ? 1 : FDE_GetCSSNameLen(psz, pEnd); |
| if (iNameLen == 0) { |
| return nullptr; |
| } |
| CFDE_CSSSelector* p = FXTARGET_NewWith(pStaticStore) |
| CFDE_CSSSelector(FDE_CSSSELECTORTYPE_Element, psz, iNameLen, true); |
| if (!p) |
| return nullptr; |
| |
| if (pFirst) { |
| pFirst->SetType(FDE_CSSSELECTORTYPE_Descendant); |
| p->SetNext(pFirst); |
| } |
| pFirst = p; |
| pLast = p; |
| psz += iNameLen; |
| } else if (wch == ':') { |
| int32_t iNameLen = FDE_GetCSSPersudoLen(psz, pEnd); |
| if (iNameLen == 0) { |
| return nullptr; |
| } |
| CFDE_CSSSelector* p = FXTARGET_NewWith(pStaticStore) |
| CFDE_CSSSelector(FDE_CSSSELECTORTYPE_Persudo, psz, iNameLen, true); |
| if (!p) |
| return nullptr; |
| |
| if (pPersudoFirst) |
| pPersudoLast->SetNext(p); |
| else |
| pPersudoFirst = p; |
| pPersudoLast = p; |
| psz += iNameLen; |
| } else if (wch == ' ') { |
| psz++; |
| } else { |
| return nullptr; |
| } |
| } |
| if (!pPersudoFirst) |
| return pFirst; |
| |
| pPersudoLast->SetNext(pFirst); |
| return pPersudoFirst; |
| } |
| |
| CFDE_CSSDeclaration* CFDE_CSSFontFaceRule::GetDeclaration() { |
| return &m_Declaration; |
| } |