Qt
Internal/Contributor docs for the Qt SDK. Note: These are NOT official API docs; those are found at https://doc.qt.io/
Loading...
Searching...
No Matches
cfx_cssdeclaration.cpp
Go to the documentation of this file.
1// Copyright 2014 The PDFium Authors
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 "core/fxcrt/css/cfx_cssdeclaration.h"
8
9#include <math.h>
10
11#include <array>
12#include <utility>
13
14#include "core/fxcrt/check.h"
15#include "core/fxcrt/check_op.h"
16#include "core/fxcrt/compiler_specific.h"
17#include "core/fxcrt/css/cfx_csscolorvalue.h"
18#include "core/fxcrt/css/cfx_csscustomproperty.h"
19#include "core/fxcrt/css/cfx_cssenumvalue.h"
20#include "core/fxcrt/css/cfx_cssnumbervalue.h"
21#include "core/fxcrt/css/cfx_csspropertyholder.h"
22#include "core/fxcrt/css/cfx_cssstringvalue.h"
23#include "core/fxcrt/css/cfx_cssvaluelist.h"
24#include "core/fxcrt/css/cfx_cssvaluelistparser.h"
25#include "core/fxcrt/fx_extension.h"
26#include "core/fxcrt/fx_system.h"
27#include "core/fxcrt/notreached.h"
28
29namespace {
30
31uint8_t Hex2Dec(uint8_t hexHigh, uint8_t hexLow) {
32 return (FXSYS_HexCharToInt(hexHigh) << 4) + FXSYS_HexCharToInt(hexLow);
33}
34
35std::optional<CFX_CSSNumber> ParseCSSNumber(WideStringView view) {
36 DCHECK(!view.IsEmpty());
37
38 size_t nUsedLen = 0;
39 float value = FXSYS_wcstof(view, &nUsedLen);
40 if (nUsedLen == 0 || !isfinite(value)) {
41 return std::nullopt;
42 }
43 view = view.Substr(nUsedLen);
44 if (view.Front() == '%') { // NOTE: empty-tolerant Front().
46 }
47 if (view.GetLength() == 2) {
48 const CFX_CSSData::LengthUnit* pUnit =
50 if (pUnit) {
51 return CFX_CSSNumber{pUnit->type, value};
52 }
53 }
55}
56
57} // namespace
58
59// static
61 WideStringView value) {
62 if (value.GetLength() >= 2) {
63 wchar_t first = value.Front();
64 wchar_t last = value.Back();
65 if ((first == '\"' && last == '\"') || (first == '\'' && last == '\'')) {
66 value = value.Substr(1, value.GetLength() - 2);
67 }
68 }
69 if (value.IsEmpty()) {
70 return std::nullopt;
71 }
72 return value;
73}
74
75// static.
77 if (value.Front() == '#') { // Note: empty-tolerant Front().
78 switch (value.GetLength()) {
79 case 4: {
80 uint8_t red = Hex2Dec((uint8_t)value[1], (uint8_t)value[1]);
81 uint8_t green = Hex2Dec((uint8_t)value[2], (uint8_t)value[2]);
82 uint8_t blue = Hex2Dec((uint8_t)value[3], (uint8_t)value[3]);
83 return ArgbEncode(255, red, green, blue);
84 }
85 case 7: {
86 uint8_t red = Hex2Dec((uint8_t)value[1], (uint8_t)value[2]);
87 uint8_t green = Hex2Dec((uint8_t)value[3], (uint8_t)value[4]);
88 uint8_t blue = Hex2Dec((uint8_t)value[5], (uint8_t)value[6]);
89 return ArgbEncode(255, red, green, blue);
90 }
91 default:
92 return std::nullopt;
93 }
94 }
95
96 if (value.GetLength() >= 10) {
97 if (!value.First(4).EqualsASCIINoCase("rgb(") || value.Back() != ')') {
98 return std::nullopt;
99 }
100 std::array<uint8_t, 3> rgb = {};
101 CFX_CSSValueListParser list(value.Substr(4, value.GetLength() - 5), ',');
102 for (auto& component : rgb) {
103 auto maybe_value = list.NextValue();
104 if (!maybe_value.has_value() ||
105 maybe_value.value().type != CFX_CSSValue::PrimitiveType::kNumber) {
106 return std::nullopt;
107 }
108 auto maybe_number = ParseCSSNumber(maybe_value.value().string_view);
109 if (!maybe_number.has_value()) {
110 return std::nullopt;
111 }
112 component = maybe_number.value().unit == CFX_CSSNumber::Unit::kPercent
113 ? FXSYS_roundf(maybe_number.value().value * 2.55f)
114 : FXSYS_roundf(maybe_number.value().value);
115 }
116 return ArgbEncode(255, rgb[0], rgb[1], rgb[2]);
117 }
118
119 const CFX_CSSData::Color* pColor = CFX_CSSData::GetColorByName(value);
120 if (!pColor) {
121 return std::nullopt;
122 }
123 return pColor->value;
124}
125
127
129
131 CFX_CSSProperty eProperty,
132 bool* bImportant) const {
133 for (const auto& p : properties_) {
134 if (p->eProperty == eProperty) {
135 *bImportant = p->bImportant;
136 return p->pValue;
137 }
138 }
139 return nullptr;
140}
141
142void CFX_CSSDeclaration::AddPropertyHolder(CFX_CSSProperty eProperty,
143 RetainPtr<CFX_CSSValue> pValue,
144 bool bImportant) {
145 auto pHolder = std::make_unique<CFX_CSSPropertyHolder>();
146 pHolder->bImportant = bImportant;
147 pHolder->eProperty = eProperty;
148 pHolder->pValue = std::move(pValue);
149 properties_.push_back(std::move(pHolder));
150}
151
153 WideStringView value) {
154 DCHECK(!value.IsEmpty());
155
156 bool bImportant = false;
157 WideStringView last_ten = value.Last(10); // NOTE: empty-tolerant Last().
158 if (last_ten.EqualsASCIINoCase("!important")) {
159 value = value.First(value.GetLength() - 10);
160 if (value.IsEmpty()) {
161 return;
162 }
163 bImportant = true;
164 }
165 const CFX_CSSValueTypeMask dwType = property->dwTypes;
166 switch (dwType & 0x0F) {
168 static constexpr CFX_CSSVALUETYPE kValueGuessOrder[] = {
173 };
174 for (CFX_CSSVALUETYPE guess : kValueGuessOrder) {
175 const CFX_CSSValueTypeMask dwMatch = dwType & guess;
176 if (dwMatch == 0) {
177 continue;
178 }
179 RetainPtr<CFX_CSSValue> pCSSValue;
180 switch (dwMatch) {
182 pCSSValue = ParseNumber(value);
183 break;
185 pCSSValue = ParseEnum(value);
186 break;
188 pCSSValue = ParseColor(value);
189 break;
191 pCSSValue = ParseString(value);
192 break;
193 default:
194 break;
195 }
196 if (pCSSValue) {
197 AddPropertyHolder(property->eName, pCSSValue, bImportant);
198 return;
199 }
200 if ((dwType & ~guess) == CFX_CSSVALUETYPE_Primitive) {
201 return;
202 }
203 }
204 break;
205 }
207 switch (property->eName) {
208 case CFX_CSSProperty::Font: {
209 ParseFontProperty(value, bImportant);
210 return;
211 }
212 case CFX_CSSProperty::Border: {
213 RetainPtr<CFX_CSSValue> pWidth = ParseBorderProperty(value);
214 AddPropertyHolder(CFX_CSSProperty::BorderLeftWidth, pWidth,
215 bImportant);
216 AddPropertyHolder(CFX_CSSProperty::BorderTopWidth, pWidth,
217 bImportant);
218 AddPropertyHolder(CFX_CSSProperty::BorderRightWidth, pWidth,
219 bImportant);
220 AddPropertyHolder(CFX_CSSProperty::BorderBottomWidth, pWidth,
221 bImportant);
222 return;
223 }
224 case CFX_CSSProperty::BorderLeft: {
225 AddPropertyHolder(CFX_CSSProperty::BorderLeftWidth,
226 ParseBorderProperty(value), bImportant);
227 break;
228 }
229 case CFX_CSSProperty::BorderTop: {
230 AddPropertyHolder(CFX_CSSProperty::BorderTopWidth,
231 ParseBorderProperty(value), bImportant);
232 return;
233 }
234 case CFX_CSSProperty::BorderRight: {
235 AddPropertyHolder(CFX_CSSProperty::BorderRightWidth,
236 ParseBorderProperty(value), bImportant);
237 return;
238 }
239 case CFX_CSSProperty::BorderBottom: {
240 AddPropertyHolder(CFX_CSSProperty::BorderBottomWidth,
241 ParseBorderProperty(value), bImportant);
242 return;
243 }
244 default:
245 break;
246 }
247 break;
248 }
250 ParseValueListProperty(property, value, bImportant);
251 return;
252 default:
254 }
255}
256
258 const WideString& value) {
259 custom_properties_.push_back(
260 std::make_unique<CFX_CSSCustomProperty>(prop, value));
261}
262
264 std::optional<CFX_CSSNumber> maybe_number = ParseCSSNumber(view);
265 if (!maybe_number.has_value()) {
266 return nullptr;
267 }
268 return pdfium::MakeRetain<CFX_CSSNumberValue>(maybe_number.value());
269}
270
272 const CFX_CSSData::PropertyValue* pValue =
274 return pValue ? pdfium::MakeRetain<CFX_CSSEnumValue>(pValue->eName) : nullptr;
275}
276
278 auto maybe_color = ParseCSSColor(value);
279 if (!maybe_color.has_value()) {
280 return nullptr;
281 }
282 return pdfium::MakeRetain<CFX_CSSColorValue>(maybe_color.value());
283}
284
286 auto maybe_string = ParseCSSString(value);
287 if (!maybe_string.has_value() || maybe_string.value().IsEmpty()) {
288 return nullptr;
289 }
290 return pdfium::MakeRetain<CFX_CSSStringValue>(maybe_string.value());
291}
292
293void CFX_CSSDeclaration::ParseValueListProperty(
294 const CFX_CSSData::Property* pProperty,
295 WideStringView value,
296 bool bImportant) {
297 wchar_t separator =
298 (pProperty->eName == CFX_CSSProperty::FontFamily) ? ',' : ' ';
299 CFX_CSSValueListParser parser(value, separator);
300 const CFX_CSSValueTypeMask dwType = pProperty->dwTypes;
301 std::vector<RetainPtr<CFX_CSSValue>> list;
302 while (1) {
303 auto maybe_next = parser.NextValue();
304 if (!maybe_next.has_value()) {
305 break;
306 }
307 switch (maybe_next.value().type) {
309 if (dwType & CFX_CSSVALUETYPE_MaybeNumber) {
310 auto maybe_number = ParseCSSNumber(maybe_next.value().string_view);
311 if (maybe_number.has_value()) {
312 list.push_back(
313 pdfium::MakeRetain<CFX_CSSNumberValue>(maybe_number.value()));
314 }
315 }
316 break;
318 if (dwType & CFX_CSSVALUETYPE_MaybeColor) {
319 auto maybe_color = ParseCSSColor(maybe_next.value().string_view);
320 if (maybe_color.has_value()) {
321 list.push_back(
322 pdfium::MakeRetain<CFX_CSSColorValue>(maybe_color.value()));
323 continue;
324 }
325 }
326 if (dwType & CFX_CSSVALUETYPE_MaybeEnum) {
327 const CFX_CSSData::PropertyValue* pPropValue =
329 maybe_next.value().string_view);
330 if (pPropValue) {
331 list.push_back(
332 pdfium::MakeRetain<CFX_CSSEnumValue>(pPropValue->eName));
333 continue;
334 }
335 }
336 if (dwType & CFX_CSSVALUETYPE_MaybeString) {
337 list.push_back(pdfium::MakeRetain<CFX_CSSStringValue>(
338 maybe_next.value().string_view));
339 }
340 break;
342 if (dwType & CFX_CSSVALUETYPE_MaybeColor) {
343 FX_ARGB color =
344 ParseCSSColor(maybe_next.value().string_view).value_or(0);
345 list.push_back(pdfium::MakeRetain<CFX_CSSColorValue>(color));
346 }
347 break;
348 default:
349 break;
350 }
351 }
352 if (list.empty()) {
353 return;
354 }
355 switch (pProperty->eName) {
356 case CFX_CSSProperty::BorderWidth:
357 Add4ValuesProperty(list, bImportant, CFX_CSSProperty::BorderLeftWidth,
358 CFX_CSSProperty::BorderTopWidth,
359 CFX_CSSProperty::BorderRightWidth,
360 CFX_CSSProperty::BorderBottomWidth);
361 return;
362 case CFX_CSSProperty::Margin:
363 Add4ValuesProperty(list, bImportant, CFX_CSSProperty::MarginLeft,
364 CFX_CSSProperty::MarginTop,
365 CFX_CSSProperty::MarginRight,
366 CFX_CSSProperty::MarginBottom);
367 return;
368 case CFX_CSSProperty::Padding:
369 Add4ValuesProperty(list, bImportant, CFX_CSSProperty::PaddingLeft,
370 CFX_CSSProperty::PaddingTop,
371 CFX_CSSProperty::PaddingRight,
372 CFX_CSSProperty::PaddingBottom);
373 return;
374 default: {
375 auto value_list = pdfium::MakeRetain<CFX_CSSValueList>(std::move(list));
376 AddPropertyHolder(pProperty->eName, std::move(value_list), bImportant);
377 return;
378 }
379 }
380}
381
382void CFX_CSSDeclaration::Add4ValuesProperty(
383 const std::vector<RetainPtr<CFX_CSSValue>>& list,
384 bool bImportant,
385 CFX_CSSProperty eLeft,
386 CFX_CSSProperty eTop,
387 CFX_CSSProperty eRight,
388 CFX_CSSProperty eBottom) {
389 switch (list.size()) {
390 case 1:
391 AddPropertyHolder(eLeft, list[0], bImportant);
392 AddPropertyHolder(eTop, list[0], bImportant);
393 AddPropertyHolder(eRight, list[0], bImportant);
394 AddPropertyHolder(eBottom, list[0], bImportant);
395 return;
396 case 2:
397 AddPropertyHolder(eLeft, list[1], bImportant);
398 AddPropertyHolder(eTop, list[0], bImportant);
399 AddPropertyHolder(eRight, list[1], bImportant);
400 AddPropertyHolder(eBottom, list[0], bImportant);
401 return;
402 case 3:
403 AddPropertyHolder(eLeft, list[1], bImportant);
404 AddPropertyHolder(eTop, list[0], bImportant);
405 AddPropertyHolder(eRight, list[1], bImportant);
406 AddPropertyHolder(eBottom, list[2], bImportant);
407 return;
408 case 4:
409 AddPropertyHolder(eLeft, list[3], bImportant);
410 AddPropertyHolder(eTop, list[0], bImportant);
411 AddPropertyHolder(eRight, list[1], bImportant);
412 AddPropertyHolder(eBottom, list[2], bImportant);
413 return;
414 default:
415 break;
416 }
417}
418
419RetainPtr<CFX_CSSValue> CFX_CSSDeclaration::ParseBorderProperty(
420 WideStringView value) const {
421 RetainPtr<CFX_CSSValue> pWidth;
422 CFX_CSSValueListParser parser(value, ' ');
423 while (1) {
424 auto maybe_next = parser.NextValue();
425 if (!maybe_next.has_value()) {
426 break;
427 }
428 switch (maybe_next.value().type) {
430 if (pWidth) {
431 continue;
432 }
433 auto maybe_number = ParseCSSNumber(maybe_next.value().string_view);
434 if (maybe_number.has_value()) {
435 pWidth = pdfium::MakeRetain<CFX_CSSNumberValue>(maybe_number.value());
436 }
437 break;
438 }
440 const CFX_CSSData::Color* pColorItem =
441 CFX_CSSData::GetColorByName(maybe_next.value().string_view);
442 if (pColorItem) {
443 continue;
444 }
445 const CFX_CSSData::PropertyValue* pValue =
446 CFX_CSSData::GetPropertyValueByName(maybe_next.value().string_view);
447 if (!pValue) {
448 continue;
449 }
450 switch (pValue->eName) {
451 case CFX_CSSPropertyValue::Thin:
452 case CFX_CSSPropertyValue::Thick:
453 case CFX_CSSPropertyValue::Medium:
454 if (!pWidth)
455 pWidth = pdfium::MakeRetain<CFX_CSSEnumValue>(pValue->eName);
456 break;
457 default:
458 break;
459 }
460 break;
461 }
462 default:
463 break;
464 }
465 }
466 if (pWidth) {
467 return pWidth;
468 }
469 return pdfium::MakeRetain<CFX_CSSNumberValue>(
471}
472
473void CFX_CSSDeclaration::ParseFontProperty(WideStringView value,
474 bool bImportant) {
475 RetainPtr<CFX_CSSValue> pStyle;
476 RetainPtr<CFX_CSSValue> pVariant;
477 RetainPtr<CFX_CSSValue> pWeight;
478 RetainPtr<CFX_CSSValue> pFontSize;
479 RetainPtr<CFX_CSSValue> pLineHeight;
480 std::vector<RetainPtr<CFX_CSSValue>> family_list;
481 CFX_CSSValueListParser parser(value, '/');
482 while (1) {
483 auto maybe_next = parser.NextValue();
484 if (!maybe_next.has_value()) {
485 break;
486 }
487 switch (maybe_next.value().type) {
489 const CFX_CSSData::PropertyValue* pValue =
490 CFX_CSSData::GetPropertyValueByName(maybe_next.value().string_view);
491 if (pValue) {
492 switch (pValue->eName) {
493 case CFX_CSSPropertyValue::XxSmall:
494 case CFX_CSSPropertyValue::XSmall:
495 case CFX_CSSPropertyValue::Small:
496 case CFX_CSSPropertyValue::Medium:
497 case CFX_CSSPropertyValue::Large:
498 case CFX_CSSPropertyValue::XLarge:
499 case CFX_CSSPropertyValue::XxLarge:
500 case CFX_CSSPropertyValue::Smaller:
501 case CFX_CSSPropertyValue::Larger:
502 if (!pFontSize)
503 pFontSize = pdfium::MakeRetain<CFX_CSSEnumValue>(pValue->eName);
504 continue;
505 case CFX_CSSPropertyValue::Bold:
506 case CFX_CSSPropertyValue::Bolder:
507 case CFX_CSSPropertyValue::Lighter:
508 if (!pWeight)
509 pWeight = pdfium::MakeRetain<CFX_CSSEnumValue>(pValue->eName);
510 continue;
511 case CFX_CSSPropertyValue::Italic:
512 case CFX_CSSPropertyValue::Oblique:
513 if (!pStyle)
514 pStyle = pdfium::MakeRetain<CFX_CSSEnumValue>(pValue->eName);
515 continue;
516 case CFX_CSSPropertyValue::SmallCaps:
517 if (!pVariant)
518 pVariant = pdfium::MakeRetain<CFX_CSSEnumValue>(pValue->eName);
519 continue;
520 case CFX_CSSPropertyValue::Normal:
521 if (!pStyle)
522 pStyle = pdfium::MakeRetain<CFX_CSSEnumValue>(pValue->eName);
523 else if (!pVariant)
524 pVariant = pdfium::MakeRetain<CFX_CSSEnumValue>(pValue->eName);
525 else if (!pWeight)
526 pWeight = pdfium::MakeRetain<CFX_CSSEnumValue>(pValue->eName);
527 else if (!pFontSize)
528 pFontSize = pdfium::MakeRetain<CFX_CSSEnumValue>(pValue->eName);
529 else if (!pLineHeight)
530 pLineHeight =
531 pdfium::MakeRetain<CFX_CSSEnumValue>(pValue->eName);
532 continue;
533 default:
534 break;
535 }
536 }
537 if (pFontSize) {
538 family_list.push_back(pdfium::MakeRetain<CFX_CSSStringValue>(
539 maybe_next.value().string_view));
540 }
542 break;
543 }
545 auto maybe_number = ParseCSSNumber(maybe_next.value().string_view);
546 if (!maybe_number.has_value()) {
547 break;
548 }
549 if (maybe_number.value().unit == CFX_CSSNumber::Unit::kNumber) {
550 switch (static_cast<int32_t>(maybe_number.value().value)) {
551 case 100:
552 case 200:
553 case 300:
554 case 400:
555 case 500:
556 case 600:
557 case 700:
558 case 800:
559 case 900:
560 if (!pWeight) {
561 pWeight = pdfium::MakeRetain<CFX_CSSNumberValue>(
562 maybe_number.value());
563 }
564 continue;
565 }
566 }
567 if (!pFontSize) {
568 pFontSize =
569 pdfium::MakeRetain<CFX_CSSNumberValue>(maybe_number.value());
570 } else if (!pLineHeight) {
571 pLineHeight =
572 pdfium::MakeRetain<CFX_CSSNumberValue>(maybe_number.value());
573 }
574 break;
575 }
576 default:
577 break;
578 }
579 }
580
581 if (!pStyle) {
582 pStyle = pdfium::MakeRetain<CFX_CSSEnumValue>(CFX_CSSPropertyValue::Normal);
583 }
584 if (!pVariant) {
585 pVariant =
586 pdfium::MakeRetain<CFX_CSSEnumValue>(CFX_CSSPropertyValue::Normal);
587 }
588 if (!pWeight) {
589 pWeight =
590 pdfium::MakeRetain<CFX_CSSEnumValue>(CFX_CSSPropertyValue::Normal);
591 }
592 if (!pFontSize) {
593 pFontSize =
594 pdfium::MakeRetain<CFX_CSSEnumValue>(CFX_CSSPropertyValue::Medium);
595 }
596 if (!pLineHeight) {
597 pLineHeight =
598 pdfium::MakeRetain<CFX_CSSEnumValue>(CFX_CSSPropertyValue::Normal);
599 }
600
601 AddPropertyHolder(CFX_CSSProperty::FontStyle, pStyle, bImportant);
602 AddPropertyHolder(CFX_CSSProperty::FontVariant, pVariant, bImportant);
603 AddPropertyHolder(CFX_CSSProperty::FontWeight, pWeight, bImportant);
604 AddPropertyHolder(CFX_CSSProperty::FontSize, pFontSize, bImportant);
605 AddPropertyHolder(CFX_CSSProperty::LineHeight, pLineHeight, bImportant);
606 if (!family_list.empty()) {
607 auto value_list =
608 pdfium::MakeRetain<CFX_CSSValueList>(std::move(family_list));
609 AddPropertyHolder(CFX_CSSProperty::FontFamily, value_list, bImportant);
610 }
611}
612
614 return properties_.size();
615}
CFX_CSSVALUETYPE
Definition cfx_css.h:14
@ CFX_CSSVALUETYPE_Shorthand
Definition cfx_css.h:17
@ CFX_CSSVALUETYPE_List
Definition cfx_css.h:16
@ CFX_CSSVALUETYPE_Primitive
Definition cfx_css.h:15
@ CFX_CSSVALUETYPE_MaybeEnum
Definition cfx_css.h:20
@ CFX_CSSVALUETYPE_MaybeString
Definition cfx_css.h:21
@ CFX_CSSVALUETYPE_MaybeColor
Definition cfx_css.h:22
@ CFX_CSSVALUETYPE_MaybeNumber
Definition cfx_css.h:19
CFX_CSSPropertyValue
Definition cfx_css.h:35
CFX_CSSProperty
Definition cfx_css.h:28
#define DCHECK
Definition check.h:33
static const Color * GetColorByName(WideStringView wsName)
static const LengthUnit * GetLengthUnitByName(WideStringView wsName)
static const PropertyValue * GetPropertyValueByName(WideStringView wsName)
void AddProperty(const CFX_CSSData::Property *property, WideStringView value)
static std::optional< WideStringView > ParseCSSString(WideStringView value)
void AddProperty(const WideString &prop, const WideString &value)
size_t PropertyCountForTesting() const
static std::optional< FX_ARGB > ParseCSSColor(WideStringView value)
RetainPtr< CFX_CSSValue > GetProperty(CFX_CSSProperty eProperty, bool *bImportant) const
std::optional< Result > NextValue()
uint32_t FX_ARGB
Definition fx_dib.h:36
constexpr FX_ARGB ArgbEncode(uint32_t a, uint32_t r, uint32_t g, uint32_t b)
Definition fx_dib.h:188
int FXSYS_HexCharToInt(char c)
#define NOTREACHED_NORETURN()
Definition notreached.h:22
fxcrt::WideStringView WideStringView
CFX_CSSNumber::Unit type
Definition cfx_cssdata.h:31
CFX_CSSPropertyValue eName
Definition cfx_cssdata.h:25
CFX_CSSProperty eName
Definition cfx_cssdata.h:19
fxcrt::WideString WideString
Definition widestring.h:207