blob: 7421f407b5ea76778a24f2c17fee66b49b180ef9 [file] [log] [blame]
// 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_cssdatatable.h"
#include "core/fxcrt/fx_ext.h"
#include "xfa/fgas/crt/fgas_codepage.h"
namespace {
uint8_t Hex2Dec(uint8_t hexHigh, uint8_t hexLow) {
return (FXSYS_toHexDigit(hexHigh) << 4) + FXSYS_toHexDigit(hexLow);
}
} // namespace
bool FDE_CSSLengthToFloat(const FDE_CSSLENGTH& len,
FX_FLOAT fPercentBase,
FX_FLOAT& fResult) {
switch (len.GetUnit()) {
case FDE_CSSLengthUnit::Point:
fResult = len.GetValue();
return true;
case FDE_CSSLengthUnit::Percent:
fResult = len.GetValue() * fPercentBase;
return true;
default:
return false;
}
}
CFX_FloatRect FDE_CSSBoundaryToRect(IFDE_CSSBoundaryStyle* pBoundStyle,
FX_FLOAT fContainerWidth,
bool bPadding,
bool bBorder,
bool bMargin) {
FX_FLOAT fResult;
const FDE_CSSRECT* pRect;
CFX_FloatRect rect(0, 0, 0, 0);
if (bPadding) {
pRect = pBoundStyle->GetPaddingWidth();
if (pRect) {
if (FDE_CSSLengthToFloat(pRect->left, fContainerWidth, fResult)) {
rect.left += fResult;
}
if (FDE_CSSLengthToFloat(pRect->top, fContainerWidth, fResult)) {
rect.top += fResult;
}
if (FDE_CSSLengthToFloat(pRect->right, fContainerWidth, fResult)) {
rect.right += fResult;
}
if (FDE_CSSLengthToFloat(pRect->bottom, fContainerWidth, fResult)) {
rect.bottom += fResult;
}
}
}
if (bBorder) {
pRect = pBoundStyle->GetBorderWidth();
if (pRect) {
if (FDE_CSSLengthToFloat(pRect->left, fContainerWidth, fResult)) {
rect.left += fResult;
}
if (FDE_CSSLengthToFloat(pRect->top, fContainerWidth, fResult)) {
rect.top += fResult;
}
if (FDE_CSSLengthToFloat(pRect->right, fContainerWidth, fResult)) {
rect.right += fResult;
}
if (FDE_CSSLengthToFloat(pRect->bottom, fContainerWidth, fResult)) {
rect.bottom += fResult;
}
}
}
if (bMargin) {
pRect = pBoundStyle->GetMarginWidth();
if (pRect) {
if (FDE_CSSLengthToFloat(pRect->left, fContainerWidth, fResult)) {
rect.left += fResult;
}
if (FDE_CSSLengthToFloat(pRect->top, fContainerWidth, fResult)) {
rect.top += fResult;
}
if (FDE_CSSLengthToFloat(pRect->right, fContainerWidth, fResult)) {
rect.right += fResult;
}
if (FDE_CSSLengthToFloat(pRect->bottom, fContainerWidth, fResult)) {
rect.bottom += fResult;
}
}
}
return rect;
}
uint32_t FDE_CSSFontStyleToFDE(IFDE_CSSFontStyle* pFontStyle) {
uint32_t dwFontStyle = FX_FONTSTYLE_Normal;
if (pFontStyle->GetFontStyle() == FDE_CSSFontStyle::Italic) {
dwFontStyle |= FX_FONTSTYLE_Italic;
}
if (pFontStyle->GetFontWeight() >= 700) {
dwFontStyle |= FX_FONTSTYLE_Bold;
}
return dwFontStyle;
}
static const FDE_CSSPropertyTable g_FDE_CSSProperties[] = {
{FDE_CSSProperty::BorderLeft, L"border-left", 0x04080036,
FDE_CSSVALUETYPE_Shorthand},
{FDE_CSSProperty::Top, L"top", 0x0BEDAF33,
FDE_CSSVALUETYPE_Primitive | FDE_CSSVALUETYPE_MaybeEnum |
FDE_CSSVALUETYPE_MaybeNumber},
{FDE_CSSProperty::Margin, L"margin", 0x0CB016BE,
FDE_CSSVALUETYPE_List | FDE_CSSVALUETYPE_MaybeEnum |
FDE_CSSVALUETYPE_MaybeNumber},
{FDE_CSSProperty::TextIndent, L"text-indent", 0x169ADB74,
FDE_CSSVALUETYPE_Primitive | FDE_CSSVALUETYPE_MaybeNumber},
{FDE_CSSProperty::Right, L"right", 0x193ADE3E,
FDE_CSSVALUETYPE_Primitive | FDE_CSSVALUETYPE_MaybeEnum |
FDE_CSSVALUETYPE_MaybeNumber},
{FDE_CSSProperty::PaddingLeft, L"padding-left", 0x228CF02F,
FDE_CSSVALUETYPE_Primitive | FDE_CSSVALUETYPE_MaybeNumber},
{FDE_CSSProperty::MarginLeft, L"margin-left", 0x297C5656,
FDE_CSSVALUETYPE_Primitive | FDE_CSSVALUETYPE_MaybeNumber |
FDE_CSSVALUETYPE_MaybeEnum},
{FDE_CSSProperty::Border, L"border", 0x2A23349E,
FDE_CSSVALUETYPE_Shorthand},
{FDE_CSSProperty::BorderTop, L"border-top", 0x2B866ADE,
FDE_CSSVALUETYPE_Shorthand},
{FDE_CSSProperty::Bottom, L"bottom", 0x399F02B5,
FDE_CSSVALUETYPE_Primitive | FDE_CSSVALUETYPE_MaybeEnum |
FDE_CSSVALUETYPE_MaybeNumber},
{FDE_CSSProperty::PaddingRight, L"padding-right", 0x3F616AC2,
FDE_CSSVALUETYPE_Primitive | FDE_CSSVALUETYPE_MaybeNumber},
{FDE_CSSProperty::BorderBottom, L"border-bottom", 0x452CE780,
FDE_CSSVALUETYPE_Shorthand},
{FDE_CSSProperty::FontFamily, L"font-family", 0x574686E6,
FDE_CSSVALUETYPE_List | FDE_CSSVALUETYPE_MaybeString},
{FDE_CSSProperty::FontWeight, L"font-weight", 0x6692F60C,
FDE_CSSVALUETYPE_Primitive | FDE_CSSVALUETYPE_MaybeEnum |
FDE_CSSVALUETYPE_MaybeNumber},
{FDE_CSSProperty::Color, L"color", 0x6E67921F,
FDE_CSSVALUETYPE_Primitive | FDE_CSSVALUETYPE_MaybeEnum |
FDE_CSSVALUETYPE_MaybeColor},
{FDE_CSSProperty::LetterSpacing, L"letter-spacing", 0x70536102,
FDE_CSSVALUETYPE_Primitive | FDE_CSSVALUETYPE_MaybeEnum |
FDE_CSSVALUETYPE_MaybeNumber},
{FDE_CSSProperty::TextAlign, L"text-align", 0x7553F1BD,
FDE_CSSVALUETYPE_Primitive | FDE_CSSVALUETYPE_MaybeEnum},
{FDE_CSSProperty::BorderRightWidth, L"border-right-width", 0x8F5A6036,
FDE_CSSVALUETYPE_Primitive | FDE_CSSVALUETYPE_MaybeEnum |
FDE_CSSVALUETYPE_MaybeNumber},
{FDE_CSSProperty::VerticalAlign, L"vertical-align", 0x934A87D2,
FDE_CSSVALUETYPE_Primitive | FDE_CSSVALUETYPE_MaybeEnum |
FDE_CSSVALUETYPE_MaybeNumber},
{FDE_CSSProperty::PaddingTop, L"padding-top", 0x959D22B7,
FDE_CSSVALUETYPE_Primitive | FDE_CSSVALUETYPE_MaybeNumber},
{FDE_CSSProperty::FontVariant, L"font-variant", 0x9C785779,
FDE_CSSVALUETYPE_Primitive | FDE_CSSVALUETYPE_MaybeEnum},
{FDE_CSSProperty::BorderWidth, L"border-width", 0xA8DE4FEB,
FDE_CSSVALUETYPE_List | FDE_CSSVALUETYPE_MaybeEnum |
FDE_CSSVALUETYPE_MaybeNumber},
{FDE_CSSProperty::BorderBottomWidth, L"border-bottom-width", 0xAE41204D,
FDE_CSSVALUETYPE_Primitive | FDE_CSSVALUETYPE_MaybeEnum |
FDE_CSSVALUETYPE_MaybeNumber},
{FDE_CSSProperty::BorderRight, L"border-right", 0xB78E9EA9,
FDE_CSSVALUETYPE_Shorthand},
{FDE_CSSProperty::FontSize, L"font-size", 0xB93956DF,
FDE_CSSVALUETYPE_Primitive | FDE_CSSVALUETYPE_MaybeEnum |
FDE_CSSVALUETYPE_MaybeNumber},
{FDE_CSSProperty::BorderSpacing, L"border-spacing", 0xC72030F0,
FDE_CSSVALUETYPE_List | FDE_CSSVALUETYPE_MaybeNumber},
{FDE_CSSProperty::FontStyle, L"font-style", 0xCB1950F5,
FDE_CSSVALUETYPE_Primitive | FDE_CSSVALUETYPE_MaybeEnum},
{FDE_CSSProperty::Font, L"font", 0xCD308B77, FDE_CSSVALUETYPE_Shorthand},
{FDE_CSSProperty::LineHeight, L"line-height", 0xCFCACE2E,
FDE_CSSVALUETYPE_Primitive | FDE_CSSVALUETYPE_MaybeEnum |
FDE_CSSVALUETYPE_MaybeNumber},
{FDE_CSSProperty::MarginRight, L"margin-right", 0xD13C58C9,
FDE_CSSVALUETYPE_Primitive | FDE_CSSVALUETYPE_MaybeNumber |
FDE_CSSVALUETYPE_MaybeEnum},
{FDE_CSSProperty::BorderLeftWidth, L"border-left-width", 0xD1E93D83,
FDE_CSSVALUETYPE_Primitive | FDE_CSSVALUETYPE_MaybeEnum |
FDE_CSSVALUETYPE_MaybeNumber},
{FDE_CSSProperty::Display, L"display", 0xD4224C36,
FDE_CSSVALUETYPE_Primitive | FDE_CSSVALUETYPE_MaybeEnum},
{FDE_CSSProperty::PaddingBottom, L"padding-bottom", 0xE555B3B9,
FDE_CSSVALUETYPE_Primitive | FDE_CSSVALUETYPE_MaybeNumber},
{FDE_CSSProperty::BorderTopWidth, L"border-top-width", 0xED2CB62B,
FDE_CSSVALUETYPE_Primitive | FDE_CSSVALUETYPE_MaybeEnum |
FDE_CSSVALUETYPE_MaybeNumber},
{FDE_CSSProperty::WordSpacing, L"word-spacing", 0xEDA63BAE,
FDE_CSSVALUETYPE_Primitive | FDE_CSSVALUETYPE_MaybeEnum |
FDE_CSSVALUETYPE_MaybeNumber},
{FDE_CSSProperty::Left, L"left", 0xF5AD782B,
FDE_CSSVALUETYPE_Primitive | FDE_CSSVALUETYPE_MaybeEnum |
FDE_CSSVALUETYPE_MaybeNumber},
{FDE_CSSProperty::TextDecoration, L"text-decoration", 0xF7C634BA,
FDE_CSSVALUETYPE_List | FDE_CSSVALUETYPE_MaybeEnum},
{FDE_CSSProperty::Padding, L"padding", 0xF8C373F7,
FDE_CSSVALUETYPE_List | FDE_CSSVALUETYPE_MaybeNumber},
{FDE_CSSProperty::MarginBottom, L"margin-bottom", 0xF93485A0,
FDE_CSSVALUETYPE_Primitive | FDE_CSSVALUETYPE_MaybeNumber |
FDE_CSSVALUETYPE_MaybeEnum},
{FDE_CSSProperty::MarginTop, L"margin-top", 0xFE51DCFE,
FDE_CSSVALUETYPE_Primitive | FDE_CSSVALUETYPE_MaybeNumber |
FDE_CSSVALUETYPE_MaybeEnum},
};
const int32_t g_iCSSPropertyCount =
sizeof(g_FDE_CSSProperties) / sizeof(FDE_CSSPropertyTable);
static_assert(g_iCSSPropertyCount ==
static_cast<int32_t>(FDE_CSSProperty::LAST_MARKER),
"Property table differs in size from property enum");
static const FDE_CSSPropertyValueTable g_FDE_CSSPropertyValues[] = {
{FDE_CSSPropertyValue::Bolder, L"bolder", 0x009F1058},
{FDE_CSSPropertyValue::None, L"none", 0x048B6670},
{FDE_CSSPropertyValue::Dot, L"dot", 0x0A48CB27},
{FDE_CSSPropertyValue::Sub, L"sub", 0x0BD37FAA},
{FDE_CSSPropertyValue::Top, L"top", 0x0BEDAF33},
{FDE_CSSPropertyValue::Right, L"right", 0x193ADE3E},
{FDE_CSSPropertyValue::Normal, L"normal", 0x247CF3E9},
{FDE_CSSPropertyValue::Auto, L"auto", 0x2B35B6D9},
{FDE_CSSPropertyValue::Text, L"text", 0x2D08AF85},
{FDE_CSSPropertyValue::XSmall, L"x-small", 0x2D2FCAFE},
{FDE_CSSPropertyValue::Thin, L"thin", 0x2D574D53},
{FDE_CSSPropertyValue::Small, L"small", 0x316A3739},
{FDE_CSSPropertyValue::Bottom, L"bottom", 0x399F02B5},
{FDE_CSSPropertyValue::Underline, L"underline", 0x3A0273A6},
{FDE_CSSPropertyValue::Double, L"double", 0x3D98515B},
{FDE_CSSPropertyValue::Lighter, L"lighter", 0x45BEB7AF},
{FDE_CSSPropertyValue::Oblique, L"oblique", 0x53EBDDB1},
{FDE_CSSPropertyValue::Super, L"super", 0x6A4F842F},
{FDE_CSSPropertyValue::Center, L"center", 0x6C51AFC1},
{FDE_CSSPropertyValue::XxLarge, L"xx-large", 0x70BB1508},
{FDE_CSSPropertyValue::Smaller, L"smaller", 0x849769F0},
{FDE_CSSPropertyValue::Baseline, L"baseline", 0x87436BA3},
{FDE_CSSPropertyValue::Thick, L"thick", 0x8CC35EB3},
{FDE_CSSPropertyValue::Justify, L"justify", 0x8D269CAE},
{FDE_CSSPropertyValue::Middle, L"middle", 0x947FA00F},
{FDE_CSSPropertyValue::Medium, L"medium", 0xA084A381},
{FDE_CSSPropertyValue::ListItem, L"list-item", 0xA32382B8},
{FDE_CSSPropertyValue::XxSmall, L"xx-small", 0xADE1FC76},
{FDE_CSSPropertyValue::Bold, L"bold", 0xB18313A1},
{FDE_CSSPropertyValue::SmallCaps, L"small-caps", 0xB299428D},
{FDE_CSSPropertyValue::Inline, L"inline", 0xC02D649F},
{FDE_CSSPropertyValue::Overline, L"overline", 0xC0EC9FA4},
{FDE_CSSPropertyValue::TextBottom, L"text-bottom", 0xC7D08D87},
{FDE_CSSPropertyValue::Larger, L"larger", 0xCD3C409D},
{FDE_CSSPropertyValue::InlineTable, L"inline-table", 0xD131F494},
{FDE_CSSPropertyValue::InlineBlock, L"inline-block", 0xD26A8BD7},
{FDE_CSSPropertyValue::Blink, L"blink", 0xDC36E390},
{FDE_CSSPropertyValue::Block, L"block", 0xDCD480AB},
{FDE_CSSPropertyValue::Italic, L"italic", 0xE31D5396},
{FDE_CSSPropertyValue::LineThrough, L"line-through", 0xE4C5A276},
{FDE_CSSPropertyValue::XLarge, L"x-large", 0xF008E390},
{FDE_CSSPropertyValue::Large, L"large", 0xF4434FCB},
{FDE_CSSPropertyValue::Left, L"left", 0xF5AD782B},
{FDE_CSSPropertyValue::TextTop, L"text-top", 0xFCB58D45},
};
const int32_t g_iCSSPropertyValueCount =
sizeof(g_FDE_CSSPropertyValues) / sizeof(FDE_CSSPropertyValueTable);
static_assert(g_iCSSPropertyValueCount ==
static_cast<int32_t>(FDE_CSSPropertyValue::LAST_MARKER),
"Property value table differs in size from property value enum");
static const FDE_CSSMEDIATYPETABLE g_FDE_CSSMediaTypes[] = {
{0xF09, FDE_CSSMEDIATYPE_Emboss}, {FDE_CSSMEDIATYPE_Screen},
{0x536A, FDE_CSSMEDIATYPE_TV}, {0x741D, FDE_CSSMEDIATYPE_Projection},
{0x76ED, FDE_CSSMEDIATYPE_Print}, {0x7CFB, FDE_CSSMEDIATYPE_Braille},
{0x9578, FDE_CSSMEDIATYPE_Handheld}, {0xC8E1, FDE_CSSMEDIATYPE_TTY},
{0xD0F9, FDE_CSSMEDIATYPE_ALL},
};
static const FDE_CSSLengthUnitTable g_FDE_CSSLengthUnits[] = {
{0x0672, FDE_CSSPrimitiveType::EMS},
{0x067D, FDE_CSSPrimitiveType::EXS},
{0x1AF7, FDE_CSSPrimitiveType::Inches},
{0x2F7A, FDE_CSSPrimitiveType::MilliMeters},
{0x3ED3, FDE_CSSPrimitiveType::Picas},
{0x3EE4, FDE_CSSPrimitiveType::Points},
{0x3EE8, FDE_CSSPrimitiveType::Pixels},
{0xFC30, FDE_CSSPrimitiveType::CentiMeters},
};
static const FDE_CSSCOLORTABLE g_FDE_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},
};
static const FDE_CSSPseudoTable g_FDE_CSSPseudoType[] = {
{FDE_CSSPseudo::After, L":after", 0x16EE1FEC},
{FDE_CSSPseudo::Before, L":before", 0x7DCDDE2D},
};
const FDE_CSSPseudoTable* FDE_GetCSSPseudoByEnum(FDE_CSSPseudo ePseudo) {
return g_FDE_CSSPseudoType + static_cast<int>(ePseudo);
}
const FDE_CSSPropertyTable* FDE_GetCSSPropertyByName(
const CFX_WideStringC& wsName) {
ASSERT(!wsName.IsEmpty());
uint32_t dwHash = FX_HashCode_GetW(wsName, true);
int32_t iEnd = g_iCSSPropertyCount;
int32_t iMid, iStart = 0;
uint32_t dwMid;
do {
iMid = (iStart + iEnd) / 2;
dwMid = g_FDE_CSSProperties[iMid].dwHash;
if (dwHash == dwMid) {
return g_FDE_CSSProperties + iMid;
} else if (dwHash > dwMid) {
iStart = iMid + 1;
} else {
iEnd = iMid - 1;
}
} while (iStart <= iEnd);
return nullptr;
}
const FDE_CSSPropertyTable* FDE_GetCSSPropertyByEnum(FDE_CSSProperty eName) {
return g_FDE_CSSProperties + static_cast<int>(eName);
}
const FDE_CSSPropertyValueTable* FDE_GetCSSPropertyValueByName(
const CFX_WideStringC& 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_FDE_CSSPropertyValues[iMid].dwHash;
if (dwHash == dwMid) {
return g_FDE_CSSPropertyValues + iMid;
} else if (dwHash > dwMid) {
iStart = iMid + 1;
} else {
iEnd = iMid - 1;
}
} while (iStart <= iEnd);
return nullptr;
}
const FDE_CSSPropertyValueTable* FDE_GetCSSPropertyValueByEnum(
FDE_CSSPropertyValue eName) {
return g_FDE_CSSPropertyValues + static_cast<int>(eName);
}
const FDE_CSSMEDIATYPETABLE* FDE_GetCSSMediaTypeByName(
const CFX_WideStringC& wsName) {
ASSERT(!wsName.IsEmpty());
uint16_t wHash = FX_HashCode_GetW(wsName, true);
int32_t iEnd =
sizeof(g_FDE_CSSMediaTypes) / sizeof(FDE_CSSMEDIATYPETABLE) - 1;
int32_t iMid, iStart = 0;
uint16_t uMid;
do {
iMid = (iStart + iEnd) / 2;
uMid = g_FDE_CSSMediaTypes[iMid].wHash;
if (wHash == uMid) {
return g_FDE_CSSMediaTypes + iMid;
} else if (wHash > uMid) {
iStart = iMid + 1;
} else {
iEnd = iMid - 1;
}
} while (iStart <= iEnd);
return nullptr;
}
const FDE_CSSLengthUnitTable* FDE_GetCSSLengthUnitByName(
const CFX_WideStringC& wsName) {
ASSERT(!wsName.IsEmpty());
uint16_t wHash = FX_HashCode_GetW(wsName, true);
int32_t iEnd =
sizeof(g_FDE_CSSLengthUnits) / sizeof(FDE_CSSLengthUnitTable) - 1;
int32_t iMid, iStart = 0;
uint16_t wMid;
do {
iMid = (iStart + iEnd) / 2;
wMid = g_FDE_CSSLengthUnits[iMid].wHash;
if (wHash == wMid) {
return g_FDE_CSSLengthUnits + iMid;
} else if (wHash > wMid) {
iStart = iMid + 1;
} else {
iEnd = iMid - 1;
}
} while (iStart <= iEnd);
return nullptr;
}
const FDE_CSSCOLORTABLE* FDE_GetCSSColorByName(const CFX_WideStringC& wsName) {
ASSERT(!wsName.IsEmpty());
uint32_t dwHash = FX_HashCode_GetW(wsName, true);
int32_t iEnd = sizeof(g_FDE_CSSColors) / sizeof(FDE_CSSCOLORTABLE) - 1;
int32_t iMid, iStart = 0;
uint32_t dwMid;
do {
iMid = (iStart + iEnd) / 2;
dwMid = g_FDE_CSSColors[iMid].dwHash;
if (dwHash == dwMid) {
return g_FDE_CSSColors + iMid;
} else if (dwHash > dwMid) {
iStart = iMid + 1;
} else {
iEnd = iMid - 1;
}
} while (iStart <= iEnd);
return nullptr;
}
bool FDE_ParseCSSNumber(const FX_WCHAR* pszValue,
int32_t iValueLen,
FX_FLOAT& fValue,
FDE_CSSPrimitiveType& 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 = FDE_CSSPrimitiveType::Number;
if (iValueLen >= 1 && *pszValue == '%') {
eUnit = FDE_CSSPrimitiveType::Percent;
} else if (iValueLen == 2) {
const FDE_CSSLengthUnitTable* pUnit =
FDE_GetCSSLengthUnitByName(CFX_WideStringC(pszValue, 2));
if (pUnit)
eUnit = pUnit->wValue;
}
return true;
}
bool FDE_ParseCSSString(const FX_WCHAR* pszValue,
int32_t iValueLen,
int32_t* iOffset,
int32_t* iLength) {
ASSERT(pszValue && iValueLen > 0);
*iOffset = 0;
*iLength = iValueLen;
if (iValueLen >= 2) {
FX_WCHAR first = pszValue[0], last = pszValue[iValueLen - 1];
if ((first == '\"' && last == '\"') || (first == '\'' && last == '\'')) {
*iOffset = 1;
*iLength -= 2;
}
}
return iValueLen > 0;
}
bool FDE_ParseCSSURI(const FX_WCHAR* pszValue,
int32_t* iOffset,
int32_t* iLength) {
ASSERT(pszValue && *iLength > 0);
if (*iLength < 6 || pszValue[*iLength - 1] != ')' ||
FXSYS_wcsnicmp(L"url(", pszValue, 4)) {
return false;
}
if (FDE_ParseCSSString(pszValue + 4, *iLength - 5, iOffset, iLength)) {
*iOffset += 4;
return true;
}
return false;
}
bool FDE_ParseCSSColor(const FX_WCHAR* pszValue,
int32_t iValueLen,
FX_ARGB& dwColor) {
ASSERT(pszValue && iValueLen > 0);
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};
FX_FLOAT fValue;
FDE_CSSPrimitiveType eType;
CFDE_CSSValueListParser list(pszValue + 4, iValueLen - 5, ',');
for (int32_t i = 0; i < 3; ++i) {
if (!list.NextValue(eType, pszValue, iValueLen))
return false;
if (eType != FDE_CSSPrimitiveType::Number)
return false;
if (!FDE_ParseCSSNumber(pszValue, iValueLen, fValue, eType))
return false;
rgb[i] = eType == FDE_CSSPrimitiveType::Percent
? FXSYS_round(fValue * 2.55f)
: FXSYS_round(fValue);
}
dwColor = ArgbEncode(255, rgb[0], rgb[1], rgb[2]);
return true;
}
const FDE_CSSCOLORTABLE* pColor =
FDE_GetCSSColorByName(CFX_WideStringC(pszValue, iValueLen));
if (!pColor)
return false;
dwColor = pColor->dwValue;
return true;
}
CFDE_CSSValueList::CFDE_CSSValueList(const CFDE_CSSValueArray& list) {
m_iCount = list.GetSize();
int32_t iByteCount = m_iCount * sizeof(IFDE_CSSValue*);
m_ppList = (IFDE_CSSValue**)FX_Alloc(uint8_t, iByteCount);
FXSYS_memcpy(m_ppList, list.GetData(), iByteCount);
}
int32_t CFDE_CSSValueList::CountValues() const {
return m_iCount;
}
IFDE_CSSValue* CFDE_CSSValueList::GetValue(int32_t index) const {
return m_ppList[index];
}
bool CFDE_CSSValueListParser::NextValue(FDE_CSSPrimitiveType& eType,
const FX_WCHAR*& pStart,
int32_t& iLength) {
while (m_pCur < m_pEnd && (*m_pCur <= ' ' || *m_pCur == m_Separator)) {
++m_pCur;
}
if (m_pCur >= m_pEnd) {
return false;
}
eType = FDE_CSSPrimitiveType::Unknown;
pStart = m_pCur;
iLength = 0;
FX_WCHAR wch = *m_pCur;
if (wch == '#') {
iLength = SkipTo(' ');
if (iLength == 4 || iLength == 7) {
eType = FDE_CSSPrimitiveType::RGB;
}
} else if ((wch >= '0' && wch <= '9') || wch == '.' || wch == '-' ||
wch == '+') {
while (m_pCur < m_pEnd && (*m_pCur > ' ' && *m_pCur != m_Separator)) {
++m_pCur;
}
iLength = m_pCur - pStart;
if (iLength > 0) {
eType = FDE_CSSPrimitiveType::Number;
}
} else if (wch == '\"' || wch == '\'') {
pStart++;
iLength = SkipTo(wch) - 1;
m_pCur++;
eType = FDE_CSSPrimitiveType::String;
} else if (m_pEnd - m_pCur > 5 && m_pCur[3] == '(') {
if (FXSYS_wcsnicmp(L"url", m_pCur, 3) == 0) {
wch = m_pCur[4];
if (wch == '\"' || wch == '\'') {
pStart += 5;
iLength = SkipTo(wch) - 6;
m_pCur += 2;
} else {
pStart += 4;
iLength = SkipTo(')') - 4;
m_pCur++;
}
eType = FDE_CSSPrimitiveType::URI;
} else if (FXSYS_wcsnicmp(L"rgb", m_pCur, 3) == 0) {
iLength = SkipTo(')') + 1;
m_pCur++;
eType = FDE_CSSPrimitiveType::RGB;
}
} else {
iLength = SkipTo(m_Separator, true, true);
eType = FDE_CSSPrimitiveType::String;
}
return m_pCur <= m_pEnd && iLength > 0;
}
int32_t CFDE_CSSValueListParser::SkipTo(FX_WCHAR wch,
bool bWSSeparator,
bool bBrContinue) {
const FX_WCHAR* pStart = m_pCur;
if (!bBrContinue) {
if (bWSSeparator) {
while ((++m_pCur < m_pEnd) && (*m_pCur != wch) && (*m_pCur > ' ')) {
continue;
}
} else {
while (++m_pCur < m_pEnd && *m_pCur != wch) {
continue;
}
}
} else {
int32_t iBracketCount = 0;
if (bWSSeparator) {
while ((m_pCur < m_pEnd) && (*m_pCur != wch) && (*m_pCur > ' ')) {
if (*m_pCur == '(') {
iBracketCount++;
} else if (*m_pCur == ')') {
iBracketCount--;
}
m_pCur++;
}
} else {
while (m_pCur < m_pEnd && *m_pCur != wch) {
if (*m_pCur == '(') {
iBracketCount++;
} else if (*m_pCur == ')') {
iBracketCount--;
}
m_pCur++;
}
}
while (iBracketCount > 0 && m_pCur < m_pEnd) {
if (*m_pCur == ')') {
iBracketCount--;
}
m_pCur++;
}
}
return m_pCur - pStart;
}
CFDE_CSSPrimitiveValue::CFDE_CSSPrimitiveValue(
const CFDE_CSSPrimitiveValue& src) = default;
CFDE_CSSPrimitiveValue::CFDE_CSSPrimitiveValue(FX_ARGB color)
: m_eType(FDE_CSSPrimitiveType::RGB), m_dwColor(color) {}
CFDE_CSSPrimitiveValue::CFDE_CSSPrimitiveValue(FDE_CSSPropertyValue eValue)
: m_eType(FDE_CSSPrimitiveType::Enum), m_eEnum(eValue) {}
CFDE_CSSPrimitiveValue::CFDE_CSSPrimitiveValue(FDE_CSSPrimitiveType eType,
FX_FLOAT fValue)
: m_eType(eType), m_fNumber(fValue) {}
CFDE_CSSPrimitiveValue::CFDE_CSSPrimitiveValue(FDE_CSSPrimitiveType eType,
const FX_WCHAR* pValue)
: m_eType(eType), m_pString(pValue) {
ASSERT(m_pString);
}
CFDE_CSSPrimitiveValue::CFDE_CSSPrimitiveValue(CFDE_CSSFunction* pFunction)
: m_eType(FDE_CSSPrimitiveType::Function), m_pFunction(pFunction) {}
FDE_CSSPrimitiveType CFDE_CSSPrimitiveValue::GetPrimitiveType() const {
return m_eType;
}
FX_ARGB CFDE_CSSPrimitiveValue::GetRGBColor() const {
ASSERT(m_eType == FDE_CSSPrimitiveType::RGB);
return m_dwColor;
}
FX_FLOAT CFDE_CSSPrimitiveValue::GetFloat() const {
ASSERT(m_eType >= FDE_CSSPrimitiveType::Number &&
m_eType <= FDE_CSSPrimitiveType::Picas);
return m_fNumber;
}
const FX_WCHAR* CFDE_CSSPrimitiveValue::GetString(int32_t& iLength) const {
ASSERT(m_eType >= FDE_CSSPrimitiveType::String &&
m_eType <= FDE_CSSPrimitiveType::URI);
iLength = FXSYS_wcslen(m_pString);
return m_pString;
}
FDE_CSSPropertyValue CFDE_CSSPrimitiveValue::GetEnum() const {
ASSERT(m_eType == FDE_CSSPrimitiveType::Enum);
return m_eEnum;
}
const FX_WCHAR* CFDE_CSSPrimitiveValue::GetFuncName() const {
ASSERT(m_eType == FDE_CSSPrimitiveType::Function);
return m_pFunction->GetFuncName();
}
int32_t CFDE_CSSPrimitiveValue::CountArgs() const {
ASSERT(m_eType == FDE_CSSPrimitiveType::Function);
return m_pFunction->CountArgs();
}
IFDE_CSSValue* CFDE_CSSPrimitiveValue::GetArgs(int32_t index) const {
ASSERT(m_eType == FDE_CSSPrimitiveType::Function);
return m_pFunction->GetArgs(index);
}