android: Update app-specific/MIME type icons
[LibreOffice.git] / include / oox / helper / helper.hxx
blob09441d11dbe1d9aee790b9b5359c123b004c5cfa
1 /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
2 /*
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 #ifndef INCLUDED_OOX_HELPER_HELPER_HXX
21 #define INCLUDED_OOX_HELPER_HELPER_HXX
23 #include <sal/config.h>
25 #include <cstring>
26 #include <limits>
27 #include <optional>
29 #include <o3tl/safeint.hxx>
30 #include <osl/endian.h>
31 #include <rtl/math.hxx>
32 #include <sal/macros.h>
33 #include <sal/types.h>
34 #include <tools/color.hxx>
35 #include <utility>
37 namespace oox {
39 // Helper macros ==============================================================
41 namespace detail {
43 //TODO: Temporary helper for STATIC_ARRAY_SELECT; ultimately, the latter should be replaced by a
44 // proper function (template):
45 template<typename T> constexpr std::make_unsigned_t<T> make_unsigned(T value) {
46 if constexpr (std::is_signed_v<T>) {
47 return o3tl::make_unsigned(value);
48 } else {
49 return value;
55 /** Expands to the 'index'-th element of a STATIC data array, or to 'def', if
56 'index' is out of the array limits. */
57 #define STATIC_ARRAY_SELECT( array, index, def ) \
58 ((detail::make_unsigned(index) < SAL_N_ELEMENTS(array)) ? ((array)[static_cast<size_t>(index)]) : (def))
60 // Common constants ===========================================================
62 const sal_uInt8 WINDOWS_CHARSET_ANSI = 0;
63 const sal_uInt8 WINDOWS_CHARSET_DEFAULT = 1;
64 const sal_uInt8 WINDOWS_CHARSET_SYMBOL = 2;
65 const sal_uInt8 WINDOWS_CHARSET_APPLE_ROMAN = 77;
66 const sal_uInt8 WINDOWS_CHARSET_SHIFTJIS = 128;
67 const sal_uInt8 WINDOWS_CHARSET_HANGEUL = 129;
68 const sal_uInt8 WINDOWS_CHARSET_JOHAB = 130;
69 const sal_uInt8 WINDOWS_CHARSET_GB2312 = 134;
70 const sal_uInt8 WINDOWS_CHARSET_BIG5 = 136;
71 const sal_uInt8 WINDOWS_CHARSET_GREEK = 161;
72 const sal_uInt8 WINDOWS_CHARSET_TURKISH = 162;
73 const sal_uInt8 WINDOWS_CHARSET_VIETNAMESE = 163;
74 const sal_uInt8 WINDOWS_CHARSET_HEBREW = 177;
75 const sal_uInt8 WINDOWS_CHARSET_ARABIC = 178;
76 const sal_uInt8 WINDOWS_CHARSET_BALTIC = 186;
77 const sal_uInt8 WINDOWS_CHARSET_RUSSIAN = 204;
78 const sal_uInt8 WINDOWS_CHARSET_THAI = 222;
79 const sal_uInt8 WINDOWS_CHARSET_EASTERN = 238;
80 const sal_uInt8 WINDOWS_CHARSET_OEM = 255;
83 const ::Color API_RGB_TRANSPARENT (ColorTransparency, 0xffffffff); ///< Transparent color for API calls.
84 const sal_uInt32 UNSIGNED_RGB_TRANSPARENT = static_cast<sal_uInt32>(-1); ///< Transparent color for unsigned int32 places.
85 const ::Color API_RGB_BLACK (0x000000); ///< Black color for API calls.
86 const ::Color API_RGB_GRAY (0x808080); ///< Gray color for API calls.
87 const ::Color API_RGB_WHITE (0xFFFFFF); ///< White color for API calls.
89 const sal_Int16 API_LINE_SOLID = 0;
90 const sal_Int16 API_LINE_DOTTED = 1;
91 const sal_Int16 API_LINE_DASHED = 2;
92 const sal_Int16 API_FINE_LINE_DASHED = 14;
94 const sal_Int16 API_LINE_NONE = 0;
95 const sal_Int16 API_LINE_HAIR = 2;
96 const sal_Int16 API_LINE_THIN = 35;
97 const sal_Int16 API_LINE_MEDIUM = 88;
98 const sal_Int16 API_LINE_THICK = 141;
100 const sal_Int16 API_ESCAPE_NONE = 0; ///< No escapement.
101 const sal_Int16 API_ESCAPE_SUPERSCRIPT = 101; ///< Superscript: raise characters automatically (magic value 101).
102 const sal_Int16 API_ESCAPE_SUBSCRIPT = -101; ///< Subscript: lower characters automatically (magic value -101).
104 const sal_Int8 API_ESCAPEHEIGHT_NONE = 100; ///< Relative character height if not escaped.
105 const sal_Int8 API_ESCAPEHEIGHT_DEFAULT = 58; ///< Relative character height if escaped.
108 // Limitate values ------------------------------------------------------------
110 template< typename ReturnType, typename Type >
111 inline ReturnType getLimitedValue( Type nValue, Type nMin, Type nMax )
113 return static_cast< ReturnType >( ::std::clamp( nValue, nMin, nMax ) );
116 template< typename ReturnType, typename Type >
117 inline ReturnType getIntervalValue( Type nValue, Type nBegin, Type nEnd )
119 static_assert(::std::numeric_limits< Type >::is_integer, "is integer");
120 Type nInterval = nEnd - nBegin;
121 Type nCount = (nValue < nBegin) ? -((nBegin - nValue - 1) / nInterval + 1) : ((nValue - nBegin) / nInterval);
122 return static_cast< ReturnType >( nValue - nCount * nInterval );
125 template< typename ReturnType >
126 inline ReturnType getDoubleIntervalValue( double fValue, double fBegin, double fEnd )
128 double fInterval = fEnd - fBegin;
129 double fCount = (fValue < fBegin) ? -(::rtl::math::approxFloor( (fBegin - fValue - 1.0) / fInterval ) + 1.0) : ::rtl::math::approxFloor( (fValue - fBegin) / fInterval );
130 return static_cast< ReturnType >( fValue - fCount * fInterval );
133 // Read from bitfields --------------------------------------------------------
135 /** Returns true, if at least one of the bits set in nMask is set in nBitField. */
136 template< typename Type >
137 inline bool getFlag( Type nBitField, Type nMask )
139 return (nBitField & nMask) != 0;
142 /** Returns nSet, if at least one bit of nMask is set in nBitField, otherwise nUnset. */
143 template< typename ReturnType, typename Type >
144 inline ReturnType getFlagValue( Type nBitField, Type nMask, ReturnType nSet, ReturnType nUnset )
146 return getFlag( nBitField, nMask ) ? nSet : nUnset;
149 /** Extracts a value from a bit field.
151 Returns the data fragment from nBitField, that starts at bit nStartBit
152 (0-based, bit 0 is rightmost) with the width of nBitCount. The returned
153 value will be right-aligned (normalized).
154 For instance: extractValue<T>(0x4321,8,4) returns 3 (value in bits 8-11).
156 template< typename ReturnType, typename Type >
157 inline ReturnType extractValue( Type nBitField, sal_uInt8 nStartBit, sal_uInt8 nBitCount )
159 sal_uInt64 nMask = 1; nMask <<= nBitCount; --nMask;
160 return static_cast< ReturnType >( nMask & (nBitField >> nStartBit) );
163 // Write to bitfields ---------------------------------------------------------
165 /** Sets or clears (according to bSet) all set bits of nMask in ornBitField. */
166 template< typename Type >
167 inline void setFlag( Type& ornBitField, Type nMask, bool bSet = true )
169 if( bSet ) ornBitField |= nMask; else ornBitField &= ~nMask;
173 template< typename Type >
174 void assignIfUsed( std::optional<Type>& rDestValue, const std::optional<Type>& rSourceValue )
176 if( rSourceValue.has_value() )
177 rDestValue = rSourceValue.value();
181 /** Provides platform independent functions to convert from or to little-endian
182 byte order, e.g. for reading data from or writing data to memory or a
183 binary stream.
185 On big-endian platforms, the byte order in the passed values is swapped,
186 this can be used for converting big-endian to and from little-endian data.
188 On little-endian platforms, the conversion functions are implemented empty,
189 thus compilers should completely optimize away the function call.
191 class ByteOrderConverter
193 public:
194 #ifdef OSL_BIGENDIAN
195 static void convertLittleEndian( sal_Int8& ) {} // present for usage in templates
196 static void convertLittleEndian( sal_uInt8& ) {} // present for usage in templates
197 static void convertLittleEndian( char16_t& rnValue ) { swap2( reinterpret_cast< sal_uInt8* >( &rnValue ) ); }
198 static void convertLittleEndian( sal_Int16& rnValue ) { swap2( reinterpret_cast< sal_uInt8* >( &rnValue ) ); }
199 static void convertLittleEndian( sal_uInt16& rnValue ) { swap2( reinterpret_cast< sal_uInt8* >( &rnValue ) ); }
200 static void convertLittleEndian( sal_Int32& rnValue ) { swap4( reinterpret_cast< sal_uInt8* >( &rnValue ) ); }
201 static void convertLittleEndian( sal_uInt32& rnValue ) { swap4( reinterpret_cast< sal_uInt8* >( &rnValue ) ); }
202 static void convertLittleEndian( sal_Int64& rnValue ) { swap8( reinterpret_cast< sal_uInt8* >( &rnValue ) ); }
203 static void convertLittleEndian( sal_uInt64& rnValue ) { swap8( reinterpret_cast< sal_uInt8* >( &rnValue ) ); }
204 static void convertLittleEndian( float& rfValue ) { swap4( reinterpret_cast< sal_uInt8* >( &rfValue ) ); }
205 static void convertLittleEndian( double& rfValue ) { swap8( reinterpret_cast< sal_uInt8* >( &rfValue ) ); }
207 template< typename Type >
208 inline static void convertLittleEndianArray( Type* pnArray, size_t nElemCount );
210 static void convertLittleEndianArray( sal_Int8*, size_t ) {}
211 static void convertLittleEndianArray( sal_uInt8*, size_t ) {}
213 #else
214 template< typename Type >
215 static void convertLittleEndian( Type& ) {}
217 template< typename Type >
218 static void convertLittleEndianArray( Type*, size_t ) {}
220 #endif
222 /** Writes a value to memory, while converting it to little-endian.
223 @param pDstBuffer The memory buffer to write the value to.
224 @param nValue The value to be written to memory in little-endian.
226 template< typename Type >
227 inline static void writeLittleEndian( void* pDstBuffer, Type nValue );
229 #ifdef OSL_BIGENDIAN
230 private:
231 inline static void swap2( sal_uInt8* pnData );
232 inline static void swap4( sal_uInt8* pnData );
233 inline static void swap8( sal_uInt8* pnData );
234 #endif
238 template< typename Type >
239 inline void ByteOrderConverter::writeLittleEndian( void* pDstBuffer, Type nValue )
241 convertLittleEndian( nValue );
242 memcpy( pDstBuffer, &nValue, sizeof( Type ) );
245 #ifdef OSL_BIGENDIAN
246 template< typename Type >
247 inline void ByteOrderConverter::convertLittleEndianArray( Type* pnArray, size_t nElemCount )
249 for( Type* pnArrayEnd = pnArray + nElemCount; pnArray != pnArrayEnd; ++pnArray )
250 convertLittleEndian( *pnArray );
253 inline void ByteOrderConverter::swap2( sal_uInt8* pnData )
255 ::std::swap( pnData[ 0 ], pnData[ 1 ] );
258 inline void ByteOrderConverter::swap4( sal_uInt8* pnData )
260 ::std::swap( pnData[ 0 ], pnData[ 3 ] );
261 ::std::swap( pnData[ 1 ], pnData[ 2 ] );
264 inline void ByteOrderConverter::swap8( sal_uInt8* pnData )
266 ::std::swap( pnData[ 0 ], pnData[ 7 ] );
267 ::std::swap( pnData[ 1 ], pnData[ 6 ] );
268 ::std::swap( pnData[ 2 ], pnData[ 5 ] );
269 ::std::swap( pnData[ 3 ], pnData[ 4 ] );
271 #endif
274 } // namespace oox
276 #endif
278 /* vim:set shiftwidth=4 softtabstop=4 expandtab: */