// 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 "core/fxcrt/css/cfx_cssdeclaration.h" #include "core/fxcrt/css/cfx_csscolorvalue.h" #include "core/fxcrt/css/cfx_csscustomproperty.h" #include "core/fxcrt/css/cfx_cssenumvalue.h" #include "core/fxcrt/css/cfx_cssnumbervalue.h" #include "core/fxcrt/css/cfx_csspropertyholder.h" #include "core/fxcrt/css/cfx_cssstringvalue.h" #include "core/fxcrt/css/cfx_cssvaluelist.h" #include "core/fxcrt/css/cfx_cssvaluelistparser.h" #include "core/fxcrt/fx_extension.h" #include "third_party/base/logging.h" #include "third_party/base/ptr_util.h" namespace { uint8_t Hex2Dec(uint8_t hexHigh, uint8_t hexLow) { return (FXSYS_HexCharToInt(hexHigh) << 4) + FXSYS_HexCharToInt(hexLow); } struct CFX_CSSPropertyValueTable { CFX_CSSPropertyValue eName; const wchar_t* pszName; uint32_t dwHash; }; const CFX_CSSPropertyValueTable g_CFX_CSSPropertyValues[] = { {CFX_CSSPropertyValue::Bolder, L"bolder", 0x009F1058}, {CFX_CSSPropertyValue::None, L"none", 0x048B6670}, {CFX_CSSPropertyValue::Dot, L"dot", 0x0A48CB27}, {CFX_CSSPropertyValue::Sub, L"sub", 0x0BD37FAA}, {CFX_CSSPropertyValue::Top, L"top", 0x0BEDAF33}, {CFX_CSSPropertyValue::Right, L"right", 0x193ADE3E}, {CFX_CSSPropertyValue::Normal, L"normal", 0x247CF3E9}, {CFX_CSSPropertyValue::Auto, L"auto", 0x2B35B6D9}, {CFX_CSSPropertyValue::Text, L"text", 0x2D08AF85}, {CFX_CSSPropertyValue::XSmall, L"x-small", 0x2D2FCAFE}, {CFX_CSSPropertyValue::Thin, L"thin", 0x2D574D53}, {CFX_CSSPropertyValue::Small, L"small", 0x316A3739}, {CFX_CSSPropertyValue::Bottom, L"bottom", 0x399F02B5}, {CFX_CSSPropertyValue::Underline, L"underline", 0x3A0273A6}, {CFX_CSSPropertyValue::Double, L"double", 0x3D98515B}, {CFX_CSSPropertyValue::Lighter, L"lighter", 0x45BEB7AF}, {CFX_CSSPropertyValue::Oblique, L"oblique", 0x53EBDDB1}, {CFX_CSSPropertyValue::Super, L"super", 0x6A4F842F}, {CFX_CSSPropertyValue::Center, L"center", 0x6C51AFC1}, {CFX_CSSPropertyValue::XxLarge, L"xx-large", 0x70BB1508}, {CFX_CSSPropertyValue::Smaller, L"smaller", 0x849769F0}, {CFX_CSSPropertyValue::Baseline, L"baseline", 0x87436BA3}, {CFX_CSSPropertyValue::Thick, L"thick", 0x8CC35EB3}, {CFX_CSSPropertyValue::Justify, L"justify", 0x8D269CAE}, {CFX_CSSPropertyValue::Middle, L"middle", 0x947FA00F}, {CFX_CSSPropertyValue::Medium, L"medium", 0xA084A381}, {CFX_CSSPropertyValue::ListItem, L"list-item", 0xA32382B8}, {CFX_CSSPropertyValue::XxSmall, L"xx-small", 0xADE1FC76}, {CFX_CSSPropertyValue::Bold, L"bold", 0xB18313A1}, {CFX_CSSPropertyValue::SmallCaps, L"small-caps", 0xB299428D}, {CFX_CSSPropertyValue::Inline, L"inline", 0xC02D649F}, {CFX_CSSPropertyValue::Overline, L"overline", 0xC0EC9FA4}, {CFX_CSSPropertyValue::TextBottom, L"text-bottom", 0xC7D08D87}, {CFX_CSSPropertyValue::Larger, L"larger", 0xCD3C409D}, {CFX_CSSPropertyValue::InlineTable, L"inline-table", 0xD131F494}, {CFX_CSSPropertyValue::InlineBlock, L"inline-block", 0xD26A8BD7}, {CFX_CSSPropertyValue::Blink, L"blink", 0xDC36E390}, {CFX_CSSPropertyValue::Block, L"block", 0xDCD480AB}, {CFX_CSSPropertyValue::Italic, L"italic", 0xE31D5396}, {CFX_CSSPropertyValue::LineThrough, L"line-through", 0xE4C5A276}, {CFX_CSSPropertyValue::XLarge, L"x-large", 0xF008E390}, {CFX_CSSPropertyValue::Large, L"large", 0xF4434FCB}, {CFX_CSSPropertyValue::Left, L"left", 0xF5AD782B}, {CFX_CSSPropertyValue::TextTop, L"text-top", 0xFCB58D45}, }; const int32_t g_iCSSPropertyValueCount = sizeof(g_CFX_CSSPropertyValues) / sizeof(CFX_CSSPropertyValueTable); static_assert(g_iCSSPropertyValueCount == static_cast<int32_t>(CFX_CSSPropertyValue::LAST_MARKER), "Property value table differs in size from property value enum"); struct CFX_CSSLengthUnitTable { uint16_t wHash; CFX_CSSNumberType wValue; }; const CFX_CSSLengthUnitTable g_CFX_CSSLengthUnits[] = { {0x0672, CFX_CSSNumberType::EMS}, {0x067D, CFX_CSSNumberType::EXS}, {0x1AF7, CFX_CSSNumberType::Inches}, {0x2F7A, CFX_CSSNumberType::MilliMeters}, {0x3ED3, CFX_CSSNumberType::Picas}, {0x3EE4, CFX_CSSNumberType::Points}, {0x3EE8, CFX_CSSNumberType::Pixels}, {0xFC30, CFX_CSSNumberType::CentiMeters}, }; struct CFX_CSSColorTable { uint32_t dwHash; FX_ARGB dwValue; }; const CFX_CSSColorTable g_CFX_CSSColors[] = { {0x031B47FE, 0xff000080}, {0x0BB8DF5B, 0xffff0000}, {0x0D82A78C, 0xff800000}, {0x2ACC82E8, 0xff00ffff}, {0x2D083986, 0xff008080}, {0x4A6A6195, 0xffc0c0c0}, {0x546A8EF3, 0xff808080}, {0x65C9169C, 0xffffa500}, {0x8422BB61, 0xffffffff}, {0x9271A558, 0xff800080}, {0xA65A3EE3, 0xffff00ff}, {0xB1345708, 0xff0000ff}, {0xB6D2CF1F, 0xff808000}, {0xD19B5E1C, 0xffffff00}, {0xDB64391D, 0xff000000}, {0xF616D507, 0xff00ff00}, {0xF6EFFF31, 0xff008000}, }; const CFX_CSSPropertyValueTable* GetCSSPropertyValueByName( const WideStringView& wsName) { ASSERT(!wsName.IsEmpty()); uint32_t dwHash = FX_HashCode_GetW(wsName, true); int32_t iEnd = g_iCSSPropertyValueCount; int32_t iMid, iStart = 0; uint32_t dwMid; do { iMid = (iStart + iEnd) / 2; dwMid = g_CFX_CSSPropertyValues[iMid].dwHash; if (dwHash == dwMid) { return g_CFX_CSSPropertyValues + iMid; } else if (dwHash > dwMid) { iStart = iMid + 1; } else { iEnd = iMid - 1; } } while (iStart <= iEnd); return nullptr; } const CFX_CSSLengthUnitTable* GetCSSLengthUnitByName( const WideStringView& wsName) { ASSERT(!wsName.IsEmpty()); uint16_t wHash = FX_HashCode_GetW(wsName, true); int32_t iEnd = sizeof(g_CFX_CSSLengthUnits) / sizeof(CFX_CSSLengthUnitTable) - 1; int32_t iMid, iStart = 0; uint16_t wMid; do { iMid = (iStart + iEnd) / 2; wMid = g_CFX_CSSLengthUnits[iMid].wHash; if (wHash == wMid) { return g_CFX_CSSLengthUnits + iMid; } else if (wHash > wMid) { iStart = iMid + 1; } else { iEnd = iMid - 1; } } while (iStart <= iEnd); return nullptr; } const CFX_CSSColorTable* GetCSSColorByName(const WideStringView& wsName) { ASSERT(!wsName.IsEmpty()); uint32_t dwHash = FX_HashCode_GetW(wsName, true); int32_t iEnd = sizeof(g_CFX_CSSColors) / sizeof(CFX_CSSColorTable) - 1; int32_t iMid, iStart = 0; uint32_t dwMid; do { iMid = (iStart + iEnd) / 2; dwMid = g_CFX_CSSColors[iMid].dwHash; if (dwHash == dwMid) { return g_CFX_CSSColors + iMid; } else if (dwHash > dwMid) { iStart = iMid + 1; } else { iEnd = iMid - 1; } } while (iStart <= iEnd); return nullptr; } bool ParseCSSNumber(const wchar_t* pszValue, int32_t iValueLen, float& fValue, CFX_CSSNumberType& eUnit) { ASSERT(pszValue && iValueLen > 0); int32_t iUsedLen = 0; fValue = FXSYS_wcstof(pszValue, iValueLen, &iUsedLen); if (iUsedLen <= 0) return false; iValueLen -= iUsedLen; pszValue += iUsedLen; eUnit = CFX_CSSNumberType::Number; if (iValueLen >= 1 && *pszValue == '%') { eUnit = CFX_CSSNumberType::Percent; } else if (iValueLen == 2) { const CFX_CSSLengthUnitTable* pUnit = GetCSSLengthUnitByName(WideStringView(pszValue, 2)); if (pUnit) eUnit = pUnit->wValue; } return true; } } // namespace // static bool CFX_CSSDeclaration::ParseCSSString(const wchar_t* pszValue, int32_t iValueLen, int32_t* iOffset, int32_t* iLength) { ASSERT(pszValue && iValueLen > 0); *iOffset = 0; *iLength = iValueLen; if (iValueLen >= 2) { wchar_t first = pszValue[0], last = pszValue[iValueLen - 1]; if ((first == '\"' && last == '\"') || (first == '\'' && last == '\'')) { *iOffset = 1; *iLength -= 2; } } return iValueLen > 0; } // static. bool CFX_CSSDeclaration::ParseCSSColor(const wchar_t* pszValue, int32_t iValueLen, FX_ARGB* dwColor) { ASSERT(pszValue && iValueLen > 0); ASSERT(dwColor); if (*pszValue == '#') { switch (iValueLen) { case 4: { uint8_t red = Hex2Dec((uint8_t)pszValue[1], (uint8_t)pszValue[1]); uint8_t green = Hex2Dec((uint8_t)pszValue[2], (uint8_t)pszValue[2]); uint8_t blue = Hex2Dec((uint8_t)pszValue[3], (uint8_t)pszValue[3]); *dwColor = ArgbEncode(255, red, green, blue); return true; } case 7: { uint8_t red = Hex2Dec((uint8_t)pszValue[1], (uint8_t)pszValue[2]); uint8_t green = Hex2Dec((uint8_t)pszValue[3], (uint8_t)pszValue[4]); uint8_t blue = Hex2Dec((uint8_t)pszValue[5], (uint8_t)pszValue[6]); *dwColor = ArgbEncode(255, red, green, blue); return true; } default: return false; } } if (iValueLen >= 10) { if (pszValue[iValueLen - 1] != ')' || FXSYS_wcsnicmp(L"rgb(", pszValue, 4)) return false; uint8_t rgb[3] = {0}; float fValue; CFX_CSSPrimitiveType eType; CFX_CSSValueListParser list(pszValue + 4, iValueLen - 5, ','); for (int32_t i = 0; i < 3; ++i) { if (!list.NextValue(&eType, &pszValue, &iValueLen)) return false; if (eType != CFX_CSSPrimitiveType::Number) return false; CFX_CSSNumberType eNumType; if (!ParseCSSNumber(pszValue, iValueLen, fValue, eNumType)) return false; rgb[i] = eNumType == CFX_CSSNumberType::Percent ? FXSYS_round(fValue * 2.55f) : FXSYS_round(fValue); } *dwColor = ArgbEncode(255, rgb[0], rgb[1], rgb[2]); return true; } const CFX_CSSColorTable* pColor = GetCSSColorByName(WideStringView(pszValue, iValueLen)); if (!pColor) return false; *dwColor = pColor->dwValue; return true; } CFX_CSSDeclaration::CFX_CSSDeclaration() {} CFX_CSSDeclaration::~CFX_CSSDeclaration() {} RetainPtr<CFX_CSSValue> CFX_CSSDeclaration::GetProperty( CFX_CSSProperty eProperty, bool* bImportant) const { for (const auto& p : properties_) { if (p->eProperty == eProperty) { *bImportant = p->bImportant; return p->pValue; } } return nullptr; } void CFX_CSSDeclaration::AddPropertyHolder(CFX_CSSProperty eProperty, RetainPtr<CFX_CSSValue> pValue, bool bImportant) { auto pHolder = pdfium::MakeUnique<CFX_CSSPropertyHolder>(); pHolder->bImportant = bImportant; pHolder->eProperty = eProperty; pHolder->pValue = pValue; properties_.push_back(std::move(pHolder)); } void CFX_CSSDeclaration::AddProperty(const CFX_CSSPropertyTable* pTable, const WideStringView& value) { ASSERT(!value.IsEmpty()); const wchar_t* pszValue = value.unterminated_c_str(); int32_t iValueLen = value.GetLength(); bool bImportant = false; if (iValueLen >= 10 && pszValue[iValueLen - 10] == '!' && FXSYS_wcsnicmp(L"important", pszValue + iValueLen - 9, 9) == 0) { if ((iValueLen -= 10) == 0) return; bImportant = true; } const uint32_t dwType = pTable->dwType; switch (dwType & 0x0F) { case CFX_CSSVALUETYPE_Primitive: { static const uint32_t g_ValueGuessOrder[] = { CFX_CSSVALUETYPE_MaybeNumber, CFX_CSSVALUETYPE_MaybeEnum, CFX_CSSVALUETYPE_MaybeColor, CFX_CSSVALUETYPE_MaybeString, }; static const int32_t g_ValueGuessCount = sizeof(g_ValueGuessOrder) / sizeof(uint32_t); for (int32_t i = 0; i < g_ValueGuessCount; ++i) { const uint32_t dwMatch = dwType & g_ValueGuessOrder[i]; if (dwMatch == 0) { continue; } RetainPtr<CFX_CSSValue> pCSSValue; switch (dwMatch) { case CFX_CSSVALUETYPE_MaybeNumber: pCSSValue = ParseNumber(pszValue, iValueLen); break; case CFX_CSSVALUETYPE_MaybeEnum: pCSSValue = ParseEnum(pszValue, iValueLen); break; case CFX_CSSVALUETYPE_MaybeColor: pCSSValue = ParseColor(pszValue, iValueLen); break; case CFX_CSSVALUETYPE_MaybeString: pCSSValue = ParseString(pszValue, iValueLen); break; default: break; } if (pCSSValue) { AddPropertyHolder(pTable->eName, pCSSValue, bImportant); return; } if ((dwType & ~(g_ValueGuessOrder[i])) == CFX_CSSVALUETYPE_Primitive) return; } break; } case CFX_CSSVALUETYPE_Shorthand: { RetainPtr<CFX_CSSValue> pWidth; switch (pTable->eName) { case CFX_CSSProperty::Font: ParseFontProperty(pszValue, iValueLen, bImportant); return; case CFX_CSSProperty::Border: if (ParseBorderProperty(pszValue, iValueLen, pWidth)) { AddPropertyHolder(CFX_CSSProperty::BorderLeftWidth, pWidth, bImportant); AddPropertyHolder(CFX_CSSProperty::BorderTopWidth, pWidth, bImportant); AddPropertyHolder(CFX_CSSProperty::BorderRightWidth, pWidth, bImportant); AddPropertyHolder(CFX_CSSProperty::BorderBottomWidth, pWidth, bImportant); return; } break; case CFX_CSSProperty::BorderLeft: if (ParseBorderProperty(pszValue, iValueLen, pWidth)) { AddPropertyHolder(CFX_CSSProperty::BorderLeftWidth, pWidth, bImportant); return; } break; case CFX_CSSProperty::BorderTop: if (ParseBorderProperty(pszValue, iValueLen, pWidth)) { AddPropertyHolder(CFX_CSSProperty::BorderTopWidth, pWidth, bImportant); return; } break; case CFX_CSSProperty::BorderRight: if (ParseBorderProperty(pszValue, iValueLen, pWidth)) { AddPropertyHolder(CFX_CSSProperty::BorderRightWidth, pWidth, bImportant); return; } break; case CFX_CSSProperty::BorderBottom: if (ParseBorderProperty(pszValue, iValueLen, pWidth)) { AddPropertyHolder(CFX_CSSProperty::BorderBottomWidth, pWidth, bImportant); return; } break; default: break; } } break; case CFX_CSSVALUETYPE_List: ParseValueListProperty(pTable, pszValue, iValueLen, bImportant); return; default: NOTREACHED(); break; } } void CFX_CSSDeclaration::AddProperty(const WideString& prop, const WideString& value) { custom_properties_.push_back( pdfium::MakeUnique<CFX_CSSCustomProperty>(prop, value)); } RetainPtr<CFX_CSSValue> CFX_CSSDeclaration::ParseNumber(const wchar_t* pszValue, int32_t iValueLen) { float fValue; CFX_CSSNumberType eUnit; if (!ParseCSSNumber(pszValue, iValueLen, fValue, eUnit)) return nullptr; return pdfium::MakeRetain<CFX_CSSNumberValue>(eUnit, fValue); } RetainPtr<CFX_CSSValue> CFX_CSSDeclaration::ParseEnum(const wchar_t* pszValue, int32_t iValueLen) { const CFX_CSSPropertyValueTable* pValue = GetCSSPropertyValueByName(WideStringView(pszValue, iValueLen)); return pValue ? pdfium::MakeRetain<CFX_CSSEnumValue>(pValue->eName) : nullptr; } RetainPtr<CFX_CSSValue> CFX_CSSDeclaration::ParseColor(const wchar_t* pszValue, int32_t iValueLen) { FX_ARGB dwColor; if (!ParseCSSColor(pszValue, iValueLen, &dwColor)) return nullptr; return pdfium::MakeRetain<CFX_CSSColorValue>(dwColor); } RetainPtr<CFX_CSSValue> CFX_CSSDeclaration::ParseString(const wchar_t* pszValue, int32_t iValueLen) { int32_t iOffset; if (!ParseCSSString(pszValue, iValueLen, &iOffset, &iValueLen)) return nullptr; if (iValueLen <= 0) return nullptr; return pdfium::MakeRetain<CFX_CSSStringValue>( WideString(pszValue + iOffset, iValueLen)); } void CFX_CSSDeclaration::ParseValueListProperty( const CFX_CSSPropertyTable* pTable, const wchar_t* pszValue, int32_t iValueLen, bool bImportant) { wchar_t separator = (pTable->eName == CFX_CSSProperty::FontFamily) ? ',' : ' '; CFX_CSSValueListParser parser(pszValue, iValueLen, separator); const uint32_t dwType = pTable->dwType; CFX_CSSPrimitiveType eType; std::vector<RetainPtr<CFX_CSSValue>> list; while (parser.NextValue(&eType, &pszValue, &iValueLen)) { switch (eType) { case CFX_CSSPrimitiveType::Number: if (dwType & CFX_CSSVALUETYPE_MaybeNumber) { float fValue; CFX_CSSNumberType eNumType; if (ParseCSSNumber(pszValue, iValueLen, fValue, eNumType)) list.push_back( pdfium::MakeRetain<CFX_CSSNumberValue>(eNumType, fValue)); } break; case CFX_CSSPrimitiveType::String: if (dwType & CFX_CSSVALUETYPE_MaybeColor) { FX_ARGB dwColor; if (ParseCSSColor(pszValue, iValueLen, &dwColor)) { list.push_back(pdfium::MakeRetain<CFX_CSSColorValue>(dwColor)); continue; } } if (dwType & CFX_CSSVALUETYPE_MaybeEnum) { const CFX_CSSPropertyValueTable* pValue = GetCSSPropertyValueByName(WideStringView(pszValue, iValueLen)); if (pValue) { list.push_back(pdfium::MakeRetain<CFX_CSSEnumValue>(pValue->eName)); continue; } } if (dwType & CFX_CSSVALUETYPE_MaybeString) { list.push_back(pdfium::MakeRetain<CFX_CSSStringValue>( WideString(pszValue, iValueLen))); } break; case CFX_CSSPrimitiveType::RGB: if (dwType & CFX_CSSVALUETYPE_MaybeColor) { FX_ARGB dwColor; if (ParseCSSColor(pszValue, iValueLen, &dwColor)) { list.push_back(pdfium::MakeRetain<CFX_CSSColorValue>(dwColor)); } } break; default: break; } } if (list.empty()) return; switch (pTable->eName) { case CFX_CSSProperty::BorderWidth: Add4ValuesProperty(list, bImportant, CFX_CSSProperty::BorderLeftWidth, CFX_CSSProperty::BorderTopWidth, CFX_CSSProperty::BorderRightWidth, CFX_CSSProperty::BorderBottomWidth); return; case CFX_CSSProperty::Margin: Add4ValuesProperty(list, bImportant, CFX_CSSProperty::MarginLeft, CFX_CSSProperty::MarginTop, CFX_CSSProperty::MarginRight, CFX_CSSProperty::MarginBottom); return; case CFX_CSSProperty::Padding: Add4ValuesProperty(list, bImportant, CFX_CSSProperty::PaddingLeft, CFX_CSSProperty::PaddingTop, CFX_CSSProperty::PaddingRight, CFX_CSSProperty::PaddingBottom); return; default: { auto pList = pdfium::MakeRetain<CFX_CSSValueList>(list); AddPropertyHolder(pTable->eName, pList, bImportant); return; } } } void CFX_CSSDeclaration::Add4ValuesProperty( const std::vector<RetainPtr<CFX_CSSValue>>& list, bool bImportant, CFX_CSSProperty eLeft, CFX_CSSProperty eTop, CFX_CSSProperty eRight, CFX_CSSProperty eBottom) { switch (list.size()) { case 1: AddPropertyHolder(eLeft, list[0], bImportant); AddPropertyHolder(eTop, list[0], bImportant); AddPropertyHolder(eRight, list[0], bImportant); AddPropertyHolder(eBottom, list[0], bImportant); return; case 2: AddPropertyHolder(eLeft, list[1], bImportant); AddPropertyHolder(eTop, list[0], bImportant); AddPropertyHolder(eRight, list[1], bImportant); AddPropertyHolder(eBottom, list[0], bImportant); return; case 3: AddPropertyHolder(eLeft, list[1], bImportant); AddPropertyHolder(eTop, list[0], bImportant); AddPropertyHolder(eRight, list[1], bImportant); AddPropertyHolder(eBottom, list[2], bImportant); return; case 4: AddPropertyHolder(eLeft, list[3], bImportant); AddPropertyHolder(eTop, list[0], bImportant); AddPropertyHolder(eRight, list[1], bImportant); AddPropertyHolder(eBottom, list[2], bImportant); return; default: break; } } bool CFX_CSSDeclaration::ParseBorderProperty( const wchar_t* pszValue, int32_t iValueLen, RetainPtr<CFX_CSSValue>& pWidth) const { pWidth.Reset(nullptr); CFX_CSSValueListParser parser(pszValue, iValueLen, ' '); CFX_CSSPrimitiveType eType; while (parser.NextValue(&eType, &pszValue, &iValueLen)) { switch (eType) { case CFX_CSSPrimitiveType::Number: { if (pWidth) continue; float fValue; CFX_CSSNumberType eNumType; if (ParseCSSNumber(pszValue, iValueLen, fValue, eNumType)) pWidth = pdfium::MakeRetain<CFX_CSSNumberValue>(eNumType, fValue); break; } case CFX_CSSPrimitiveType::String: { const CFX_CSSColorTable* pColorItem = GetCSSColorByName(WideStringView(pszValue, iValueLen)); if (pColorItem) continue; const CFX_CSSPropertyValueTable* pValue = GetCSSPropertyValueByName(WideStringView(pszValue, iValueLen)); if (!pValue) continue; switch (pValue->eName) { case CFX_CSSPropertyValue::Thin: case CFX_CSSPropertyValue::Thick: case CFX_CSSPropertyValue::Medium: if (!pWidth) pWidth = pdfium::MakeRetain<CFX_CSSEnumValue>(pValue->eName); break; default: break; } break; } default: break; } } if (!pWidth) pWidth = pdfium::MakeRetain<CFX_CSSNumberValue>(CFX_CSSNumberType::Number, 0.0f); return true; } void CFX_CSSDeclaration::ParseFontProperty(const wchar_t* pszValue, int32_t iValueLen, bool bImportant) { CFX_CSSValueListParser parser(pszValue, iValueLen, '/'); RetainPtr<CFX_CSSValue> pStyle; RetainPtr<CFX_CSSValue> pVariant; RetainPtr<CFX_CSSValue> pWeight; RetainPtr<CFX_CSSValue> pFontSize; RetainPtr<CFX_CSSValue> pLineHeight; std::vector<RetainPtr<CFX_CSSValue>> familyList; CFX_CSSPrimitiveType eType; while (parser.NextValue(&eType, &pszValue, &iValueLen)) { switch (eType) { case CFX_CSSPrimitiveType::String: { const CFX_CSSPropertyValueTable* pValue = GetCSSPropertyValueByName(WideStringView(pszValue, iValueLen)); if (pValue) { switch (pValue->eName) { case CFX_CSSPropertyValue::XxSmall: case CFX_CSSPropertyValue::XSmall: case CFX_CSSPropertyValue::Small: case CFX_CSSPropertyValue::Medium: case CFX_CSSPropertyValue::Large: case CFX_CSSPropertyValue::XLarge: case CFX_CSSPropertyValue::XxLarge: case CFX_CSSPropertyValue::Smaller: case CFX_CSSPropertyValue::Larger: if (!pFontSize) pFontSize = pdfium::MakeRetain<CFX_CSSEnumValue>(pValue->eName); continue; case CFX_CSSPropertyValue::Bold: case CFX_CSSPropertyValue::Bolder: case CFX_CSSPropertyValue::Lighter: if (!pWeight) pWeight = pdfium::MakeRetain<CFX_CSSEnumValue>(pValue->eName); continue; case CFX_CSSPropertyValue::Italic: case CFX_CSSPropertyValue::Oblique: if (!pStyle) pStyle = pdfium::MakeRetain<CFX_CSSEnumValue>(pValue->eName); continue; case CFX_CSSPropertyValue::SmallCaps: if (!pVariant) pVariant = pdfium::MakeRetain<CFX_CSSEnumValue>(pValue->eName); continue; case CFX_CSSPropertyValue::Normal: if (!pStyle) pStyle = pdfium::MakeRetain<CFX_CSSEnumValue>(pValue->eName); else if (!pVariant) pVariant = pdfium::MakeRetain<CFX_CSSEnumValue>(pValue->eName); else if (!pWeight) pWeight = pdfium::MakeRetain<CFX_CSSEnumValue>(pValue->eName); else if (!pFontSize) pFontSize = pdfium::MakeRetain<CFX_CSSEnumValue>(pValue->eName); else if (!pLineHeight) pLineHeight = pdfium::MakeRetain<CFX_CSSEnumValue>(pValue->eName); continue; default: break; } } if (pFontSize) { familyList.push_back(pdfium::MakeRetain<CFX_CSSStringValue>( WideString(pszValue, iValueLen))); } parser.UseCommaSeparator(); break; } case CFX_CSSPrimitiveType::Number: { float fValue; CFX_CSSNumberType eNumType; if (!ParseCSSNumber(pszValue, iValueLen, fValue, eNumType)) break; if (eType == CFX_CSSPrimitiveType::Number) { switch ((int32_t)fValue) { case 100: case 200: case 300: case 400: case 500: case 600: case 700: case 800: case 900: if (!pWeight) pWeight = pdfium::MakeRetain<CFX_CSSNumberValue>( CFX_CSSNumberType::Number, fValue); continue; } } if (!pFontSize) pFontSize = pdfium::MakeRetain<CFX_CSSNumberValue>(eNumType, fValue); else if (!pLineHeight) pLineHeight = pdfium::MakeRetain<CFX_CSSNumberValue>(eNumType, fValue); break; } default: break; } } if (!pStyle) { pStyle = pdfium::MakeRetain<CFX_CSSEnumValue>(CFX_CSSPropertyValue::Normal); } if (!pVariant) { pVariant = pdfium::MakeRetain<CFX_CSSEnumValue>(CFX_CSSPropertyValue::Normal); } if (!pWeight) { pWeight = pdfium::MakeRetain<CFX_CSSEnumValue>(CFX_CSSPropertyValue::Normal); } if (!pFontSize) { pFontSize = pdfium::MakeRetain<CFX_CSSEnumValue>(CFX_CSSPropertyValue::Medium); } if (!pLineHeight) { pLineHeight = pdfium::MakeRetain<CFX_CSSEnumValue>(CFX_CSSPropertyValue::Normal); } AddPropertyHolder(CFX_CSSProperty::FontStyle, pStyle, bImportant); AddPropertyHolder(CFX_CSSProperty::FontVariant, pVariant, bImportant); AddPropertyHolder(CFX_CSSProperty::FontWeight, pWeight, bImportant); AddPropertyHolder(CFX_CSSProperty::FontSize, pFontSize, bImportant); AddPropertyHolder(CFX_CSSProperty::LineHeight, pLineHeight, bImportant); if (!familyList.empty()) { auto pList = pdfium::MakeRetain<CFX_CSSValueList>(familyList); AddPropertyHolder(CFX_CSSProperty::FontFamily, pList, bImportant); } } size_t CFX_CSSDeclaration::PropertyCountForTesting() const { return properties_.size(); }