| OLD | NEW |
| (Empty) |
| 1 // Copyright 2014 PDFium Authors. All rights reserved. | |
| 2 // Use of this source code is governed by a BSD-style license that can be | |
| 3 // found in the LICENSE file. | |
| 4 | |
| 5 // Original code copyright 2014 Foxit Software Inc. http://www.foxitsoftware.com | |
| 6 | |
| 7 #include "xfa/fxfa/parser/xfa_utils.h" | |
| 8 | |
| 9 #include "core/fxcrt/include/fx_ext.h" | |
| 10 #include "xfa/fde/xml/fde_xml_imp.h" | |
| 11 #include "xfa/fxfa/parser/cxfa_document.h" | |
| 12 #include "xfa/fxfa/parser/cxfa_measurement.h" | |
| 13 #include "xfa/fxfa/parser/xfa_basic_data.h" | |
| 14 #include "xfa/fxfa/parser/xfa_localemgr.h" | |
| 15 #include "xfa/fxfa/parser/xfa_localevalue.h" | |
| 16 #include "xfa/fxfa/parser/xfa_object.h" | |
| 17 | |
| 18 namespace { | |
| 19 | |
| 20 const FX_DOUBLE fraction_scales[] = {0.1, | |
| 21 0.01, | |
| 22 0.001, | |
| 23 0.0001, | |
| 24 0.00001, | |
| 25 0.000001, | |
| 26 0.0000001, | |
| 27 0.00000001, | |
| 28 0.000000001, | |
| 29 0.0000000001, | |
| 30 0.00000000001, | |
| 31 0.000000000001, | |
| 32 0.0000000000001, | |
| 33 0.00000000000001, | |
| 34 0.000000000000001, | |
| 35 0.0000000000000001}; | |
| 36 | |
| 37 FX_DOUBLE WideStringToDouble(const CFX_WideString& wsStringVal) { | |
| 38 CFX_WideString wsValue = wsStringVal; | |
| 39 wsValue.TrimLeft(); | |
| 40 wsValue.TrimRight(); | |
| 41 int64_t nIntegral = 0; | |
| 42 uint32_t dwFractional = 0; | |
| 43 int32_t nExponent = 0; | |
| 44 int32_t cc = 0; | |
| 45 bool bNegative = false; | |
| 46 bool bExpSign = false; | |
| 47 const FX_WCHAR* str = wsValue.c_str(); | |
| 48 int32_t len = wsValue.GetLength(); | |
| 49 if (str[0] == '+') { | |
| 50 cc++; | |
| 51 } else if (str[0] == '-') { | |
| 52 bNegative = true; | |
| 53 cc++; | |
| 54 } | |
| 55 int32_t nIntegralLen = 0; | |
| 56 while (cc < len) { | |
| 57 if (str[cc] == '.' || str[cc] == 'E' || str[cc] == 'e' || | |
| 58 nIntegralLen > 17) { | |
| 59 break; | |
| 60 } | |
| 61 if (!FXSYS_isDecimalDigit(str[cc])) { | |
| 62 return 0; | |
| 63 } | |
| 64 nIntegral = nIntegral * 10 + str[cc] - '0'; | |
| 65 cc++; | |
| 66 nIntegralLen++; | |
| 67 } | |
| 68 nIntegral = bNegative ? -nIntegral : nIntegral; | |
| 69 int32_t scale = 0; | |
| 70 FX_DOUBLE fraction = 0.0; | |
| 71 if (cc < len && str[cc] == '.') { | |
| 72 cc++; | |
| 73 while (cc < len) { | |
| 74 fraction += fraction_scales[scale] * (str[cc] - '0'); | |
| 75 scale++; | |
| 76 cc++; | |
| 77 if (cc == len) { | |
| 78 break; | |
| 79 } | |
| 80 if (scale == sizeof(fraction_scales) / sizeof(FX_DOUBLE) || | |
| 81 str[cc] == 'E' || str[cc] == 'e') { | |
| 82 break; | |
| 83 } | |
| 84 if (!FXSYS_isDecimalDigit(str[cc])) { | |
| 85 return 0; | |
| 86 } | |
| 87 } | |
| 88 dwFractional = (uint32_t)(fraction * 4294967296.0); | |
| 89 } | |
| 90 if (cc < len && (str[cc] == 'E' || str[cc] == 'e')) { | |
| 91 cc++; | |
| 92 if (cc < len) { | |
| 93 if (str[cc] == '+') { | |
| 94 cc++; | |
| 95 } else if (str[cc] == '-') { | |
| 96 bExpSign = true; | |
| 97 cc++; | |
| 98 } | |
| 99 } | |
| 100 while (cc < len) { | |
| 101 if (str[cc] == '.' || !FXSYS_isDecimalDigit(str[cc])) { | |
| 102 return 0; | |
| 103 } | |
| 104 nExponent = nExponent * 10 + str[cc] - '0'; | |
| 105 cc++; | |
| 106 } | |
| 107 nExponent = bExpSign ? -nExponent : nExponent; | |
| 108 } | |
| 109 FX_DOUBLE dValue = (dwFractional / 4294967296.0); | |
| 110 dValue = nIntegral + (nIntegral >= 0 ? dValue : -dValue); | |
| 111 if (nExponent != 0) { | |
| 112 dValue *= FXSYS_pow(10, (FX_FLOAT)nExponent); | |
| 113 } | |
| 114 return dValue; | |
| 115 } | |
| 116 | |
| 117 } // namespace | |
| 118 | |
| 119 CXFA_LocaleValue XFA_GetLocaleValue(CXFA_WidgetData* pWidgetData) { | |
| 120 CXFA_Node* pNodeValue = | |
| 121 pWidgetData->GetNode()->GetChild(0, XFA_Element::Value); | |
| 122 if (!pNodeValue) { | |
| 123 return CXFA_LocaleValue(); | |
| 124 } | |
| 125 CXFA_Node* pValueChild = pNodeValue->GetNodeItem(XFA_NODEITEM_FirstChild); | |
| 126 if (!pValueChild) { | |
| 127 return CXFA_LocaleValue(); | |
| 128 } | |
| 129 int32_t iVTType = XFA_VT_NULL; | |
| 130 switch (pValueChild->GetElementType()) { | |
| 131 case XFA_Element::Decimal: | |
| 132 iVTType = XFA_VT_DECIMAL; | |
| 133 break; | |
| 134 case XFA_Element::Float: | |
| 135 iVTType = XFA_VT_FLOAT; | |
| 136 break; | |
| 137 case XFA_Element::Date: | |
| 138 iVTType = XFA_VT_DATE; | |
| 139 break; | |
| 140 case XFA_Element::Time: | |
| 141 iVTType = XFA_VT_TIME; | |
| 142 break; | |
| 143 case XFA_Element::DateTime: | |
| 144 iVTType = XFA_VT_DATETIME; | |
| 145 break; | |
| 146 case XFA_Element::Boolean: | |
| 147 iVTType = XFA_VT_BOOLEAN; | |
| 148 break; | |
| 149 case XFA_Element::Integer: | |
| 150 iVTType = XFA_VT_INTEGER; | |
| 151 break; | |
| 152 case XFA_Element::Text: | |
| 153 iVTType = XFA_VT_TEXT; | |
| 154 break; | |
| 155 default: | |
| 156 iVTType = XFA_VT_NULL; | |
| 157 break; | |
| 158 } | |
| 159 return CXFA_LocaleValue(iVTType, pWidgetData->GetRawValue(), | |
| 160 pWidgetData->GetNode()->GetDocument()->GetLocalMgr()); | |
| 161 } | |
| 162 void XFA_GetPlainTextFromRichText(CFDE_XMLNode* pXMLNode, | |
| 163 CFX_WideString& wsPlainText) { | |
| 164 if (!pXMLNode) { | |
| 165 return; | |
| 166 } | |
| 167 switch (pXMLNode->GetType()) { | |
| 168 case FDE_XMLNODE_Element: { | |
| 169 CFDE_XMLElement* pXMLElement = static_cast<CFDE_XMLElement*>(pXMLNode); | |
| 170 CFX_WideString wsTag; | |
| 171 pXMLElement->GetLocalTagName(wsTag); | |
| 172 uint32_t uTag = FX_HashCode_GetW(wsTag.AsStringC(), true); | |
| 173 if (uTag == 0x0001f714) { | |
| 174 wsPlainText += L"\n"; | |
| 175 } else if (uTag == 0x00000070) { | |
| 176 if (!wsPlainText.IsEmpty()) { | |
| 177 wsPlainText += L"\n"; | |
| 178 } | |
| 179 } else if (uTag == 0xa48ac63) { | |
| 180 if (!wsPlainText.IsEmpty() && | |
| 181 wsPlainText[wsPlainText.GetLength() - 1] != '\n') { | |
| 182 wsPlainText += L"\n"; | |
| 183 } | |
| 184 } | |
| 185 } break; | |
| 186 case FDE_XMLNODE_Text: { | |
| 187 CFX_WideString wsContent; | |
| 188 static_cast<CFDE_XMLText*>(pXMLNode)->GetText(wsContent); | |
| 189 wsPlainText += wsContent; | |
| 190 } break; | |
| 191 case FDE_XMLNODE_CharData: { | |
| 192 CFX_WideString wsCharData; | |
| 193 static_cast<CFDE_XMLCharData*>(pXMLNode)->GetCharData(wsCharData); | |
| 194 wsPlainText += wsCharData; | |
| 195 } break; | |
| 196 default: | |
| 197 break; | |
| 198 } | |
| 199 for (CFDE_XMLNode* pChildXML = | |
| 200 pXMLNode->GetNodeItem(CFDE_XMLNode::FirstChild); | |
| 201 pChildXML; | |
| 202 pChildXML = pChildXML->GetNodeItem(CFDE_XMLNode::NextSibling)) { | |
| 203 XFA_GetPlainTextFromRichText(pChildXML, wsPlainText); | |
| 204 } | |
| 205 } | |
| 206 | |
| 207 FX_BOOL XFA_FieldIsMultiListBox(CXFA_Node* pFieldNode) { | |
| 208 FX_BOOL bRet = FALSE; | |
| 209 if (!pFieldNode) | |
| 210 return bRet; | |
| 211 | |
| 212 CXFA_Node* pUIChild = pFieldNode->GetChild(0, XFA_Element::Ui); | |
| 213 if (pUIChild) { | |
| 214 CXFA_Node* pFirstChild = pUIChild->GetNodeItem(XFA_NODEITEM_FirstChild); | |
| 215 if (pFirstChild && | |
| 216 pFirstChild->GetElementType() == XFA_Element::ChoiceList) { | |
| 217 bRet = pFirstChild->GetEnum(XFA_ATTRIBUTE_Open) == | |
| 218 XFA_ATTRIBUTEENUM_MultiSelect; | |
| 219 } | |
| 220 } | |
| 221 return bRet; | |
| 222 } | |
| 223 | |
| 224 FX_DOUBLE XFA_ByteStringToDouble(const CFX_ByteStringC& szStringVal) { | |
| 225 CFX_WideString wsValue = CFX_WideString::FromUTF8(szStringVal); | |
| 226 return WideStringToDouble(wsValue); | |
| 227 } | |
| 228 | |
| 229 int32_t XFA_MapRotation(int32_t nRotation) { | |
| 230 nRotation = nRotation % 360; | |
| 231 nRotation = nRotation < 0 ? nRotation + 360 : nRotation; | |
| 232 return nRotation; | |
| 233 } | |
| 234 | |
| 235 const XFA_SCRIPTATTRIBUTEINFO* XFA_GetScriptAttributeByName( | |
| 236 XFA_Element eElement, | |
| 237 const CFX_WideStringC& wsAttributeName) { | |
| 238 if (wsAttributeName.IsEmpty()) | |
| 239 return nullptr; | |
| 240 | |
| 241 int32_t iElementIndex = static_cast<int32_t>(eElement); | |
| 242 while (iElementIndex != -1) { | |
| 243 const XFA_SCRIPTHIERARCHY* scriptIndex = g_XFAScriptIndex + iElementIndex; | |
| 244 int32_t icount = scriptIndex->wAttributeCount; | |
| 245 if (icount == 0) { | |
| 246 iElementIndex = scriptIndex->wParentIndex; | |
| 247 continue; | |
| 248 } | |
| 249 uint32_t uHash = FX_HashCode_GetW(wsAttributeName, false); | |
| 250 int32_t iStart = scriptIndex->wAttributeStart, iEnd = iStart + icount - 1; | |
| 251 do { | |
| 252 int32_t iMid = (iStart + iEnd) / 2; | |
| 253 const XFA_SCRIPTATTRIBUTEINFO* pInfo = g_SomAttributeData + iMid; | |
| 254 if (uHash == pInfo->uHash) | |
| 255 return pInfo; | |
| 256 if (uHash < pInfo->uHash) | |
| 257 iEnd = iMid - 1; | |
| 258 else | |
| 259 iStart = iMid + 1; | |
| 260 } while (iStart <= iEnd); | |
| 261 iElementIndex = scriptIndex->wParentIndex; | |
| 262 } | |
| 263 return nullptr; | |
| 264 } | |
| 265 | |
| 266 const XFA_NOTSUREATTRIBUTE* XFA_GetNotsureAttribute(XFA_Element eElement, | |
| 267 XFA_ATTRIBUTE eAttribute, | |
| 268 XFA_ATTRIBUTETYPE eType) { | |
| 269 int32_t iStart = 0, iEnd = g_iXFANotsureCount - 1; | |
| 270 do { | |
| 271 int32_t iMid = (iStart + iEnd) / 2; | |
| 272 const XFA_NOTSUREATTRIBUTE* pAttr = g_XFANotsureAttributes + iMid; | |
| 273 if (eElement == pAttr->eElement) { | |
| 274 if (pAttr->eAttribute == eAttribute) { | |
| 275 if (eType == XFA_ATTRIBUTETYPE_NOTSURE || eType == pAttr->eType) | |
| 276 return pAttr; | |
| 277 return nullptr; | |
| 278 } | |
| 279 int32_t iBefore = iMid - 1; | |
| 280 if (iBefore >= 0) { | |
| 281 pAttr = g_XFANotsureAttributes + iBefore; | |
| 282 while (eElement == pAttr->eElement) { | |
| 283 if (pAttr->eAttribute == eAttribute) { | |
| 284 if (eType == XFA_ATTRIBUTETYPE_NOTSURE || eType == pAttr->eType) | |
| 285 return pAttr; | |
| 286 return nullptr; | |
| 287 } | |
| 288 iBefore--; | |
| 289 if (iBefore < 0) | |
| 290 break; | |
| 291 | |
| 292 pAttr = g_XFANotsureAttributes + iBefore; | |
| 293 } | |
| 294 } | |
| 295 | |
| 296 int32_t iAfter = iMid + 1; | |
| 297 if (iAfter <= g_iXFANotsureCount - 1) { | |
| 298 pAttr = g_XFANotsureAttributes + iAfter; | |
| 299 while (eElement == pAttr->eElement) { | |
| 300 if (pAttr->eAttribute == eAttribute) { | |
| 301 if (eType == XFA_ATTRIBUTETYPE_NOTSURE || eType == pAttr->eType) | |
| 302 return pAttr; | |
| 303 return nullptr; | |
| 304 } | |
| 305 iAfter++; | |
| 306 if (iAfter > g_iXFANotsureCount - 1) | |
| 307 break; | |
| 308 | |
| 309 pAttr = g_XFANotsureAttributes + iAfter; | |
| 310 } | |
| 311 } | |
| 312 return nullptr; | |
| 313 } | |
| 314 | |
| 315 if (eElement < pAttr->eElement) | |
| 316 iEnd = iMid - 1; | |
| 317 else | |
| 318 iStart = iMid + 1; | |
| 319 } while (iStart <= iEnd); | |
| 320 return nullptr; | |
| 321 } | |
| 322 | |
| 323 const XFA_PROPERTY* XFA_GetPropertyOfElement(XFA_Element eElement, | |
| 324 XFA_Element eProperty, | |
| 325 uint32_t dwPacket) { | |
| 326 int32_t iCount = 0; | |
| 327 const XFA_PROPERTY* pProperties = XFA_GetElementProperties(eElement, iCount); | |
| 328 if (!pProperties || iCount < 1) | |
| 329 return nullptr; | |
| 330 | |
| 331 auto it = std::find_if(pProperties, pProperties + iCount, | |
| 332 [eProperty](const XFA_PROPERTY& prop) { | |
| 333 return prop.eName == eProperty; | |
| 334 }); | |
| 335 if (it == pProperties + iCount) | |
| 336 return nullptr; | |
| 337 | |
| 338 const XFA_ELEMENTINFO* pInfo = XFA_GetElementByID(eProperty); | |
| 339 ASSERT(pInfo); | |
| 340 if (dwPacket != XFA_XDPPACKET_UNKNOWN && !(dwPacket & pInfo->dwPackets)) | |
| 341 return nullptr; | |
| 342 return it; | |
| 343 } | |
| 344 | |
| 345 const XFA_PROPERTY* XFA_GetElementProperties(XFA_Element eElement, | |
| 346 int32_t& iCount) { | |
| 347 if (eElement == XFA_Element::Unknown) | |
| 348 return nullptr; | |
| 349 | |
| 350 const XFA_ELEMENTHIERARCHY* pElement = | |
| 351 g_XFAElementPropertyIndex + static_cast<int32_t>(eElement); | |
| 352 iCount = pElement->wCount; | |
| 353 return g_XFAElementPropertyData + pElement->wStart; | |
| 354 } | |
| 355 | |
| 356 const uint8_t* XFA_GetElementAttributes(XFA_Element eElement, int32_t& iCount) { | |
| 357 if (eElement == XFA_Element::Unknown) | |
| 358 return nullptr; | |
| 359 | |
| 360 const XFA_ELEMENTHIERARCHY* pElement = | |
| 361 g_XFAElementAttributeIndex + static_cast<int32_t>(eElement); | |
| 362 iCount = pElement->wCount; | |
| 363 return g_XFAElementAttributeData + pElement->wStart; | |
| 364 } | |
| 365 | |
| 366 const XFA_ELEMENTINFO* XFA_GetElementByID(XFA_Element eName) { | |
| 367 return eName != XFA_Element::Unknown | |
| 368 ? g_XFAElementData + static_cast<int32_t>(eName) | |
| 369 : nullptr; | |
| 370 } | |
| 371 | |
| 372 XFA_Element XFA_GetElementTypeForName(const CFX_WideStringC& wsName) { | |
| 373 if (wsName.IsEmpty()) | |
| 374 return XFA_Element::Unknown; | |
| 375 | |
| 376 uint32_t uHash = FX_HashCode_GetW(wsName, false); | |
| 377 const XFA_ELEMENTINFO* pEnd = g_XFAElementData + g_iXFAElementCount; | |
| 378 auto pInfo = std::lower_bound(g_XFAElementData, pEnd, uHash, | |
| 379 [](const XFA_ELEMENTINFO& info, uint32_t hash) { | |
| 380 return info.uHash < hash; | |
| 381 }); | |
| 382 if (pInfo < pEnd && pInfo->uHash == uHash) | |
| 383 return pInfo->eName; | |
| 384 return XFA_Element::Unknown; | |
| 385 } | |
| 386 | |
| 387 CXFA_Measurement XFA_GetAttributeDefaultValue_Measure(XFA_Element eElement, | |
| 388 XFA_ATTRIBUTE eAttribute, | |
| 389 uint32_t dwPacket) { | |
| 390 void* pValue; | |
| 391 if (XFA_GetAttributeDefaultValue(pValue, eElement, eAttribute, | |
| 392 XFA_ATTRIBUTETYPE_Measure, dwPacket)) { | |
| 393 return *(CXFA_Measurement*)pValue; | |
| 394 } | |
| 395 return CXFA_Measurement(); | |
| 396 } | |
| 397 | |
| 398 FX_BOOL XFA_GetAttributeDefaultValue(void*& pValue, | |
| 399 XFA_Element eElement, | |
| 400 XFA_ATTRIBUTE eAttribute, | |
| 401 XFA_ATTRIBUTETYPE eType, | |
| 402 uint32_t dwPacket) { | |
| 403 const XFA_ATTRIBUTEINFO* pInfo = XFA_GetAttributeByID(eAttribute); | |
| 404 if (!pInfo) | |
| 405 return FALSE; | |
| 406 if (dwPacket && (dwPacket & pInfo->dwPackets) == 0) | |
| 407 return FALSE; | |
| 408 if (pInfo->eType == eType) { | |
| 409 pValue = pInfo->pDefValue; | |
| 410 return TRUE; | |
| 411 } | |
| 412 if (pInfo->eType == XFA_ATTRIBUTETYPE_NOTSURE) { | |
| 413 const XFA_NOTSUREATTRIBUTE* pAttr = | |
| 414 XFA_GetNotsureAttribute(eElement, eAttribute, eType); | |
| 415 if (pAttr) { | |
| 416 pValue = pAttr->pValue; | |
| 417 return TRUE; | |
| 418 } | |
| 419 } | |
| 420 return FALSE; | |
| 421 } | |
| 422 | |
| 423 const XFA_ATTRIBUTEINFO* XFA_GetAttributeByName(const CFX_WideStringC& wsName) { | |
| 424 if (wsName.IsEmpty()) | |
| 425 return nullptr; | |
| 426 | |
| 427 uint32_t uHash = FX_HashCode_GetW(wsName, false); | |
| 428 int32_t iStart = 0; | |
| 429 int32_t iEnd = g_iXFAAttributeCount - 1; | |
| 430 do { | |
| 431 int32_t iMid = (iStart + iEnd) / 2; | |
| 432 const XFA_ATTRIBUTEINFO* pInfo = g_XFAAttributeData + iMid; | |
| 433 if (uHash == pInfo->uHash) | |
| 434 return pInfo; | |
| 435 if (uHash < pInfo->uHash) | |
| 436 iEnd = iMid - 1; | |
| 437 else | |
| 438 iStart = iMid + 1; | |
| 439 } while (iStart <= iEnd); | |
| 440 return nullptr; | |
| 441 } | |
| 442 | |
| 443 const XFA_ATTRIBUTEINFO* XFA_GetAttributeByID(XFA_ATTRIBUTE eName) { | |
| 444 return (eName < g_iXFAAttributeCount) ? (g_XFAAttributeData + eName) | |
| 445 : nullptr; | |
| 446 } | |
| 447 | |
| 448 const XFA_ATTRIBUTEENUMINFO* XFA_GetAttributeEnumByName( | |
| 449 const CFX_WideStringC& wsName) { | |
| 450 if (wsName.IsEmpty()) | |
| 451 return nullptr; | |
| 452 | |
| 453 uint32_t uHash = FX_HashCode_GetW(wsName, false); | |
| 454 int32_t iStart = 0; | |
| 455 int32_t iEnd = g_iXFAEnumCount - 1; | |
| 456 do { | |
| 457 int32_t iMid = (iStart + iEnd) / 2; | |
| 458 const XFA_ATTRIBUTEENUMINFO* pInfo = g_XFAEnumData + iMid; | |
| 459 if (uHash == pInfo->uHash) | |
| 460 return pInfo; | |
| 461 if (uHash < pInfo->uHash) | |
| 462 iEnd = iMid - 1; | |
| 463 else | |
| 464 iStart = iMid + 1; | |
| 465 } while (iStart <= iEnd); | |
| 466 return nullptr; | |
| 467 } | |
| 468 | |
| 469 const XFA_PACKETINFO* XFA_GetPacketByIndex(XFA_PACKET ePacket) { | |
| 470 return g_XFAPacketData + ePacket; | |
| 471 } | |
| 472 | |
| 473 const XFA_PACKETINFO* XFA_GetPacketByID(uint32_t dwPacket) { | |
| 474 int32_t iStart = 0, iEnd = g_iXFAPacketCount - 1; | |
| 475 do { | |
| 476 int32_t iMid = (iStart + iEnd) / 2; | |
| 477 uint32_t dwFind = (g_XFAPacketData + iMid)->eName; | |
| 478 if (dwPacket == dwFind) | |
| 479 return g_XFAPacketData + iMid; | |
| 480 if (dwPacket < dwFind) | |
| 481 iEnd = iMid - 1; | |
| 482 else | |
| 483 iStart = iMid + 1; | |
| 484 } while (iStart <= iEnd); | |
| 485 return nullptr; | |
| 486 } | |
| OLD | NEW |