1 /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
3 * This file is part of the LibreOffice project.
5 * This Source Code Form is subject to the terms of the Mozilla Public
6 * License, v. 2.0. If a copy of the MPL was not distributed with this
7 * file, You can obtain one at http://mozilla.org/MPL/2.0/.
9 * This file incorporates work covered by the following license notice:
11 * Licensed to the Apache Software Foundation (ASF) under one or more
12 * contributor license agreements. See the NOTICE file distributed
13 * with this work for additional information regarding copyright
14 * ownership. The ASF licenses this file to you under the Apache
15 * License, Version 2.0 (the "License"); you may not use this file
16 * except in compliance with the License. You may obtain a copy of
17 * the License at http://www.apache.org/licenses/LICENSE-2.0 .
20 #include "XMLNumberStylesExport.hxx"
21 #include <XMLNumberStylesImport.hxx>
22 #include <xmloff/xmlnamespace.hxx>
23 #include <xmloff/xmlimp.hxx>
24 #include <xmloff/namespacemap.hxx>
25 #include <xmloff/xmltoken.hxx>
27 #include <sal/log.hxx>
29 #include "sdxmlexp_impl.hxx"
30 #include "sdxmlimp_impl.hxx"
32 using namespace ::xmloff::token
;
36 struct SdXMLDataStyleNumber
38 enum XMLTokenEnum meNumberStyle
;
47 SdXMLDataStyleNumber
const aSdXMLDataStyleNumbers
[] =
49 { XML_DAY
, false, false, false, nullptr },
50 { XML_DAY
, true, false, false, nullptr },
51 { XML_MONTH
, true, false, false, nullptr },
52 { XML_MONTH
, false, true, false, nullptr },
53 { XML_MONTH
, true, true, false, nullptr },
54 { XML_YEAR
, false, false, false, nullptr },
55 { XML_YEAR
, true, false, false, nullptr },
56 { XML_DAY_OF_WEEK
, false, false, false, nullptr },
57 { XML_DAY_OF_WEEK
, true, false, false, nullptr },
58 { XML_TEXT
, false, false, false, "." },
59 { XML_TEXT
, false, false, false, " " },
60 { XML_TEXT
, false, false, false, ", " },
61 { XML_TEXT
, false, false, false, ". " },
62 { XML_HOURS
, false, false, false, nullptr },
63 { XML_MINUTES
, false, false, false, nullptr },
64 { XML_TEXT
, false, false, false, ":" },
65 { XML_AM_PM
, false, false, false, nullptr },
66 { XML_SECONDS
, false, false, false, nullptr },
67 { XML_SECONDS
, false, false, true, nullptr },
68 { XML_TOKEN_INVALID
, false, false, false, nullptr }
72 enum class DataStyleNumber
: sal_uInt8
75 Day
= 1, // <number:day/>
76 DayLong
= 2, // <number:day number:style="long"/>
77 MonthLong
= 3, // <number:month number:style="long"/>
78 MonthText
= 4, // <number:month number:textual="true"/>
79 MonthLongText
= 5, // <number:month number:style="long" number:textual="true"/>
80 Year
= 6, // <number:year/>
81 YearLong
= 7, // <number:year number:style="long"/>
82 DayOfWeek
= 8, // <number:day-of-week/>
83 DayOfWeekLong
= 9, // <number:day-of-week number:style="long"/>
84 TextPoint
= 10, // <number:text>.</number:text>
85 TextSpace
= 11, // <number:text> </number:text>
86 TextCommaSpace
= 12, // <number:text>, </number:text>
87 TextPointSpace
= 13, // <number:text>. </number:text>
88 Hours
= 14, // <number:hours/>
89 Minutes
= 15, // <number:minutes/>
90 TextColon
= 16, // <number:text>:</number:text>
91 AmPm
= 17, // <number:am-pm/>
92 Seconds
= 18, // <number:seconds/>
93 Seconds_02
= 19, // <number:seconds number:/>
96 struct SdXMLFixedDataStyle
101 DataStyleNumber mpFormat
[8];
104 const SdXMLFixedDataStyle aSdXML_Standard_Short
=
108 DataStyleNumber::DayLong
,
109 DataStyleNumber::TextPoint
,
110 DataStyleNumber::MonthLong
,
111 DataStyleNumber::TextPoint
,
112 DataStyleNumber::YearLong
,
113 DataStyleNumber::NONE
, DataStyleNumber::NONE
, DataStyleNumber::NONE
117 const SdXMLFixedDataStyle aSdXML_Standard_Long
=
121 DataStyleNumber::DayOfWeekLong
,
122 DataStyleNumber::TextCommaSpace
,
123 DataStyleNumber::Day
,
124 DataStyleNumber::TextPointSpace
,
125 DataStyleNumber::MonthLongText
,
126 DataStyleNumber::TextSpace
,
127 DataStyleNumber::YearLong
,
128 DataStyleNumber::NONE
132 const SdXMLFixedDataStyle aSdXML_DateStyle_1
=
136 DataStyleNumber::DayLong
,
137 DataStyleNumber::TextPoint
,
138 DataStyleNumber::MonthLong
,
139 DataStyleNumber::TextPoint
,
140 DataStyleNumber::Year
,
141 DataStyleNumber::NONE
, DataStyleNumber::NONE
, DataStyleNumber::NONE
145 const SdXMLFixedDataStyle aSdXML_DateStyle_2
=
149 DataStyleNumber::DayLong
,
150 DataStyleNumber::TextPoint
,
151 DataStyleNumber::MonthLong
,
152 DataStyleNumber::TextPoint
,
153 DataStyleNumber::YearLong
,
154 DataStyleNumber::NONE
, DataStyleNumber::NONE
, DataStyleNumber::NONE
158 const SdXMLFixedDataStyle aSdXML_DateStyle_3
=
162 DataStyleNumber::Day
,
163 DataStyleNumber::TextPointSpace
,
164 DataStyleNumber::MonthText
,
165 DataStyleNumber::TextSpace
,
166 DataStyleNumber::YearLong
,
167 DataStyleNumber::NONE
, DataStyleNumber::NONE
, DataStyleNumber::NONE
171 const SdXMLFixedDataStyle aSdXML_DateStyle_4
=
175 DataStyleNumber::Day
,
176 DataStyleNumber::TextPointSpace
,
177 DataStyleNumber::MonthLongText
,
178 DataStyleNumber::TextSpace
,
179 DataStyleNumber::YearLong
,
180 DataStyleNumber::NONE
, DataStyleNumber::NONE
, DataStyleNumber::NONE
184 const SdXMLFixedDataStyle aSdXML_DateStyle_5
=
188 DataStyleNumber::DayOfWeek
,
189 DataStyleNumber::TextCommaSpace
,
190 DataStyleNumber::Day
,
191 DataStyleNumber::TextPointSpace
,
192 DataStyleNumber::MonthLongText
,
193 DataStyleNumber::TextSpace
,
194 DataStyleNumber::YearLong
,
195 DataStyleNumber::NONE
199 const SdXMLFixedDataStyle aSdXML_DateStyle_6
=
203 DataStyleNumber::DayOfWeekLong
,
204 DataStyleNumber::TextCommaSpace
,
205 DataStyleNumber::Day
,
206 DataStyleNumber::TextPointSpace
,
207 DataStyleNumber::MonthLongText
,
208 DataStyleNumber::TextSpace
,
209 DataStyleNumber::YearLong
,
210 DataStyleNumber::NONE
214 const SdXMLFixedDataStyle aSdXML_TimeStyle_1
=
217 DataStyleNumber::Hours
,
218 DataStyleNumber::TextColon
,
219 DataStyleNumber::Minutes
,
220 DataStyleNumber::TextColon
,
221 DataStyleNumber::Seconds
,
222 DataStyleNumber::AmPm
,
223 DataStyleNumber::NONE
, DataStyleNumber::NONE
227 const SdXMLFixedDataStyle aSdXML_TimeStyle_2
=
228 { "T2", false, false,
230 DataStyleNumber::Hours
,
231 DataStyleNumber::TextColon
,
232 DataStyleNumber::Minutes
,
233 DataStyleNumber::NONE
, DataStyleNumber::NONE
, DataStyleNumber::NONE
, DataStyleNumber::NONE
, DataStyleNumber::NONE
237 const SdXMLFixedDataStyle aSdXML_TimeStyle_3
=
238 { "T3", false, false,
240 DataStyleNumber::Hours
,
241 DataStyleNumber::TextColon
,
242 DataStyleNumber::Minutes
,
243 DataStyleNumber::TextColon
,
244 DataStyleNumber::Seconds
,
245 DataStyleNumber::NONE
, DataStyleNumber::NONE
, DataStyleNumber::NONE
249 const SdXMLFixedDataStyle aSdXML_TimeStyle_4
=
250 { "T4", false, false,
252 DataStyleNumber::Hours
,
253 DataStyleNumber::TextColon
,
254 DataStyleNumber::Minutes
,
255 DataStyleNumber::TextColon
,
256 DataStyleNumber::Seconds_02
,
257 DataStyleNumber::NONE
, DataStyleNumber::NONE
, DataStyleNumber::NONE
261 const SdXMLFixedDataStyle aSdXML_TimeStyle_5
=
262 { "T5", false, false,
264 DataStyleNumber::Hours
,
265 DataStyleNumber::TextColon
,
266 DataStyleNumber::Minutes
,
267 DataStyleNumber::AmPm
,
268 DataStyleNumber::NONE
, DataStyleNumber::NONE
, DataStyleNumber::NONE
, DataStyleNumber::NONE
272 const SdXMLFixedDataStyle aSdXML_TimeStyle_6
=
273 { "T6", false, false,
275 DataStyleNumber::Hours
,
276 DataStyleNumber::TextColon
,
277 DataStyleNumber::Minutes
,
278 DataStyleNumber::TextColon
,
279 DataStyleNumber::Seconds
,
280 DataStyleNumber::AmPm
,
281 DataStyleNumber::NONE
, DataStyleNumber::NONE
285 const SdXMLFixedDataStyle aSdXML_TimeStyle_7
=
286 { "T7", false, false,
288 DataStyleNumber::Hours
,
289 DataStyleNumber::TextColon
,
290 DataStyleNumber::Minutes
,
291 DataStyleNumber::TextColon
,
292 DataStyleNumber::Seconds_02
,
293 DataStyleNumber::AmPm
,
294 DataStyleNumber::NONE
, DataStyleNumber::NONE
298 const SdXMLFixedDataStyle
* const aSdXMLFixedDateFormats
[SdXMLDateFormatCount
] =
300 &aSdXML_Standard_Short
,
301 &aSdXML_Standard_Long
,
310 const SdXMLFixedDataStyle
* const aSdXMLFixedTimeFormats
[SdXMLTimeFormatCount
] =
323 static void SdXMLExportDataStyleNumber( SdXMLExport
& rExport
, SdXMLDataStyleNumber
const & rElement
)
325 if( rElement
.mbDecimal02
)
327 rExport
.AddAttribute( XML_NAMESPACE_NUMBER
, XML_DECIMAL_PLACES
, XML_2
);
330 if( rElement
.mbLong
)
332 rExport
.AddAttribute( XML_NAMESPACE_NUMBER
, XML_STYLE
, XML_LONG
);
335 if( rElement
.mbTextual
)
337 rExport
.AddAttribute( XML_NAMESPACE_NUMBER
, XML_TEXTUAL
, XML_TRUE
);
340 SvXMLElementExport
aNumberStyle( rExport
, XML_NAMESPACE_NUMBER
, rElement
.meNumberStyle
, true, false );
341 if( rElement
.mpText
)
343 OUString
sAttrValue( OUString::createFromAscii( rElement
.mpText
) );
344 rExport
.GetDocHandler()->characters( sAttrValue
);
348 static void SdXMLExportStyle( SdXMLExport
& rExport
, const SdXMLFixedDataStyle
* pStyle
, const SdXMLFixedDataStyle
* pStyle2
= nullptr )
351 OUString sAttrValue
= OUString::createFromAscii( pStyle
->mpName
);
353 sAttrValue
+= OUString::createFromAscii( pStyle2
->mpName
);
355 rExport
.AddAttribute( XML_NAMESPACE_STYLE
, XML_NAME
, sAttrValue
);
357 if( pStyle
->mbAutomatic
)
359 rExport
.AddAttribute( XML_NAMESPACE_NUMBER
, XML_AUTOMATIC_ORDER
, XML_TRUE
);
362 SvXMLElementExport
aElement( rExport
, XML_NAMESPACE_NUMBER
, pStyle
->mbDateStyle
? XML_DATE_STYLE
: XML_TIME_STYLE
, true, true );
367 const DataStyleNumber
* pElements
= &pStyle
->mpFormat
[0];
369 while( *pElements
!= DataStyleNumber::NONE
)
371 SdXMLDataStyleNumber
const & rElement
= aSdXMLDataStyleNumbers
[ static_cast<int>(*pElements
++) - 1 ];
372 SdXMLExportDataStyleNumber( rExport
, rElement
);
377 SdXMLDataStyleNumber
const & rElement
= aSdXMLDataStyleNumbers
[ static_cast<int>(DataStyleNumber::TextSpace
) - 1 ];
378 SdXMLExportDataStyleNumber( rExport
, rElement
);
387 void SdXMLNumberStylesExporter::exportTimeStyle( SdXMLExport
& rExport
, sal_Int32 nStyle
)
389 SAL_WARN_IF( (nStyle
< 0) || (nStyle
>= SdXMLTimeFormatCount
), "xmloff", "Unknown time style!" );
390 if( (nStyle
>= 0) && (nStyle
< SdXMLTimeFormatCount
) )
391 SdXMLExportStyle( rExport
, aSdXMLFixedTimeFormats
[ nStyle
] );
394 void SdXMLNumberStylesExporter::exportDateStyle( SdXMLExport
& rExport
, sal_Int32 nStyle
)
398 int nDateStyle
= nStyle
& 0x0f;
399 bool bHasDate
= nDateStyle
!= 0;
404 SAL_WARN_IF(nDateStyle
>= SdXMLDateFormatCount
, "xmloff", "unknown date style!");
406 int nTimeStyle
= (nStyle
>> 4) & 0x0f;
407 bool bHasTime
= nTimeStyle
!= 0;
412 SAL_WARN_IF(nTimeStyle
>= SdXMLTimeFormatCount
, "xmloff", "Unknown time style!");
414 if ((nDateStyle
< SdXMLDateFormatCount
) && (nTimeStyle
< SdXMLTimeFormatCount
))
420 SdXMLExportStyle( rExport
, aSdXMLFixedDateFormats
[ nDateStyle
], aSdXMLFixedTimeFormats
[ nTimeStyle
] );
424 SdXMLExportStyle( rExport
, aSdXMLFixedDateFormats
[ nDateStyle
] );
429 SdXMLExportStyle( rExport
, aSdXMLFixedTimeFormats
[ nTimeStyle
] );
435 SAL_WARN_IF( (nStyle
< 0) || (nStyle
>= SdXMLDateFormatCount
), "xmloff", "unknown date style!" );
436 if( (nStyle
>= 0) && (nStyle
< SdXMLDateFormatCount
) )
437 SdXMLExportStyle( rExport
, aSdXMLFixedDateFormats
[ nStyle
] );
441 OUString
SdXMLNumberStylesExporter::getTimeStyleName(const sal_Int32 nTimeFormat
)
443 sal_Int32 nFormat
= nTimeFormat
;
447 if( (nFormat
>= 0) && (nFormat
< SdXMLTimeFormatCount
) )
449 return OUString::createFromAscii(aSdXMLFixedTimeFormats
[nFormat
]->mpName
);
457 OUString
SdXMLNumberStylesExporter::getDateStyleName(const sal_Int32 nDateFormat
)
459 sal_Int32 nFormat
= nDateFormat
;
465 aStr
= getDateStyleName( nFormat
& 0x0f );
466 aStr
+= getTimeStyleName( (nFormat
>> 4) & 0x0f );
473 if( (nFormat
>= 0) && (nFormat
< SdXMLDateFormatCount
) )
475 return OUString::createFromAscii(aSdXMLFixedDateFormats
[nFormat
]->mpName
);
485 class SdXMLNumberFormatMemberImportContext
: public SvXMLImportContext
488 SdXMLNumberFormatImportContext
* mpParent
;
490 OUString maNumberStyle
;
495 css::uno::Reference
< css::xml::sax::XFastContextHandler
> mxSlaveContext
;
499 SdXMLNumberFormatMemberImportContext( SvXMLImport
& rImport
,
501 const css::uno::Reference
< css::xml::sax::XFastAttributeList
>& xAttrList
,
502 SdXMLNumberFormatImportContext
* pParent
,
503 const css::uno::Reference
< css::xml::sax::XFastContextHandler
>& rSlaveContext
);
505 virtual css::uno::Reference
< css::xml::sax::XFastContextHandler
> SAL_CALL
createFastChildContext(
506 sal_Int32 nElement
, const css::uno::Reference
< css::xml::sax::XFastAttributeList
>& AttrList
) override
;
508 virtual void SAL_CALL
startFastElement( sal_Int32 nElement
,
509 const css::uno::Reference
< css::xml::sax::XFastAttributeList
>& ) override
;
511 virtual void SAL_CALL
endFastElement(sal_Int32 nElement
) override
;
513 virtual void SAL_CALL
characters( const OUString
& rChars
) override
;
517 SdXMLNumberFormatMemberImportContext::SdXMLNumberFormatMemberImportContext(
518 SvXMLImport
& rImport
,
520 const css::uno::Reference
< css::xml::sax::XFastAttributeList
>& xAttrList
,
521 SdXMLNumberFormatImportContext
* pParent
,
522 const css::uno::Reference
< css::xml::sax::XFastContextHandler
>& rSlaveContext
)
523 : SvXMLImportContext(rImport
),
525 maNumberStyle( SvXMLImport::getNameFromToken(nElement
) ),
526 mxSlaveContext( rSlaveContext
)
532 for (auto &aIter
: sax_fastparser::castToFastAttributeList( xAttrList
))
534 OUString sValue
= aIter
.toString();
536 switch (aIter
.getToken())
538 case XML_ELEMENT(NUMBER
, XML_DECIMAL_PLACES
):
539 mbDecimal02
= IsXMLToken( sValue
, XML_2
);
541 case XML_ELEMENT(NUMBER
, XML_STYLE
):
542 mbLong
= IsXMLToken( sValue
, XML_LONG
);
544 case XML_ELEMENT(NUMBER
, XML_TEXTUAL
):
545 mbTextual
= IsXMLToken( sValue
, XML_TRUE
);
548 XMLOFF_WARN_UNKNOWN("xmloff", aIter
);
554 css::uno::Reference
< css::xml::sax::XFastContextHandler
> SdXMLNumberFormatMemberImportContext::createFastChildContext(
556 const css::uno::Reference
< css::xml::sax::XFastAttributeList
>& xAttrList
)
558 return mxSlaveContext
->createFastChildContext( nElement
, xAttrList
);
561 void SdXMLNumberFormatMemberImportContext::startFastElement(
563 const css::uno::Reference
< css::xml::sax::XFastAttributeList
>& xAttrList
)
565 mxSlaveContext
->startFastElement( nElement
, xAttrList
);
568 void SdXMLNumberFormatMemberImportContext::endFastElement(sal_Int32 nElement
)
570 mxSlaveContext
->endFastElement(nElement
);
573 mpParent
->add( maNumberStyle
, mbLong
, mbTextual
, mbDecimal02
, maText
);
576 void SdXMLNumberFormatMemberImportContext::characters( const OUString
& rChars
)
578 mxSlaveContext
->characters( rChars
);
583 SdXMLNumberFormatImportContext::SdXMLNumberFormatImportContext( SdXMLImport
& rImport
, sal_Int32 nElement
, SvXMLNumImpData
* pNewData
, SvXMLStylesTokens nNewType
, const css::uno::Reference
< css::xml::sax::XFastAttributeList
>& xAttrList
, SvXMLStylesContext
& rStyles
)
584 : SvXMLNumFormatContext(rImport
, nElement
, pNewData
, nNewType
, xAttrList
, rStyles
),
585 mbAutomatic( false ),
590 mbTimeStyle
= (nElement
& TOKEN_MASK
) == XML_TIME_STYLE
;
592 for (auto &aIter
: sax_fastparser::castToFastAttributeList( xAttrList
))
594 OUString sValue
= aIter
.toString();
595 if( aIter
.getToken() == XML_ELEMENT(NUMBER
, XML_AUTOMATIC_ORDER
) )
596 mbAutomatic
= IsXMLToken( sValue
, XML_TRUE
);
598 XMLOFF_WARN_UNKNOWN("xmloff", aIter
);
602 SdXMLNumberFormatImportContext::~SdXMLNumberFormatImportContext()
606 void SdXMLNumberFormatImportContext::add( OUString
const & rNumberStyle
, bool bLong
, bool bTextual
, bool bDecimal02
, OUString
const & rText
)
611 const SdXMLDataStyleNumber
* pStyleMember
= aSdXMLDataStyleNumbers
;
612 for( sal_uInt8 nIndex
= 0; pStyleMember
->meNumberStyle
!= XML_TOKEN_INVALID
; nIndex
++, pStyleMember
++ )
614 if( IsXMLToken(rNumberStyle
, pStyleMember
->meNumberStyle
) &&
615 (pStyleMember
->mbLong
== bLong
) &&
616 (pStyleMember
->mbTextual
== bTextual
) &&
617 (pStyleMember
->mbDecimal02
== bDecimal02
) &&
618 ( ( (pStyleMember
->mpText
== nullptr) && (rText
.isEmpty()) ) ||
619 ( pStyleMember
->mpText
&& (rText
.equalsAscii( pStyleMember
->mpText
) ) ) ) )
621 mnElements
[mnIndex
++] = static_cast<DataStyleNumber
>(nIndex
+ 1);
627 bool SdXMLNumberFormatImportContext::compareStyle( const SdXMLFixedDataStyle
* pStyle
, sal_Int16
& nIndex
) const
629 if( (pStyle
->mbAutomatic
!= mbAutomatic
) && (nIndex
== 0))
632 sal_Int16 nCompareIndex
;
633 for( nCompareIndex
= 0; nCompareIndex
< 8; nIndex
++, nCompareIndex
++ )
635 if( pStyle
->mpFormat
[nCompareIndex
] != mnElements
[nIndex
] )
642 void SdXMLNumberFormatImportContext::endFastElement(sal_Int32
)
644 for( ; mnIndex
< 16; mnIndex
++ )
646 mnElements
[mnIndex
] = DataStyleNumber::NONE
;
651 // compare import with all time styles
652 for( sal_Int16 nFormat
= 0; nFormat
< SdXMLTimeFormatCount
; nFormat
++ )
654 sal_Int16 nIndex
= 0;
655 if( compareStyle( aSdXMLFixedTimeFormats
[nFormat
], nIndex
) )
664 // compare import with all date styles
665 for( sal_Int16 nFormat
= 0; nFormat
< SdXMLDateFormatCount
; nFormat
++ )
667 sal_Int16 nIndex
= 0;
668 if( compareStyle( aSdXMLFixedDateFormats
[nFormat
], nIndex
) )
673 else if( mnElements
[nIndex
] == DataStyleNumber::TextSpace
)
675 // if it's a valid date ending with a space, see if a time style follows
676 for( sal_Int16 nTimeFormat
= 0; nTimeFormat
< SdXMLTimeFormatCount
; nTimeFormat
++ )
678 sal_Int16 nIndex2
= nIndex
+ 1;
679 if( compareStyle( aSdXMLFixedTimeFormats
[nTimeFormat
], nIndex2
) )
681 mnKey
= (nFormat
+ 2) | ((nTimeFormat
+ 2) << 4);
688 // no date style found? maybe it's an extended time style
691 // compare import with all time styles
692 for( sal_Int16 nFormat
= 0; nFormat
< SdXMLTimeFormatCount
; nFormat
++ )
694 sal_Int16 nIndex
= 0;
695 if( compareStyle( aSdXMLFixedTimeFormats
[nFormat
], nIndex
) )
697 mnKey
= (nFormat
+ 2) << 4;
705 css::uno::Reference
< css::xml::sax::XFastContextHandler
> SdXMLNumberFormatImportContext::createFastChildContext(
707 const css::uno::Reference
< css::xml::sax::XFastAttributeList
>& xAttrList
)
709 return new SdXMLNumberFormatMemberImportContext( GetImport(), nElement
, xAttrList
,
710 this, SvXMLNumFormatContext::createFastChildContext( nElement
, xAttrList
) );
713 /* vim:set shiftwidth=4 softtabstop=4 expandtab: */