nss: upgrade to release 3.73
[LibreOffice.git] / scaddins / source / datefunc / datefunc.cxx
blob995db62353008b43e7da91c3d43be06654d0fe8d
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 #include "datefunc.hxx"
21 #include <datefunc.hrc>
22 #include <strings.hrc>
23 #include <com/sun/star/util/Date.hpp>
24 #include <cppuhelper/factory.hxx>
25 #include <cppuhelper/supportsservice.hxx>
26 #include <cppuhelper/weak.hxx>
27 #include <com/sun/star/lang/XSingleServiceFactory.hpp>
28 #include <rtl/ustrbuf.hxx>
29 #include <unotools/resmgr.hxx>
30 #include <i18nlangtag/languagetag.hxx>
31 #include <algorithm>
32 #include "deffuncname.hxx"
34 using namespace ::com::sun::star;
36 #define ADDIN_SERVICE "com.sun.star.sheet.AddIn"
37 #define MY_SERVICE "com.sun.star.sheet.addin.DateFunctions"
38 #define MY_IMPLNAME "com.sun.star.sheet.addin.DateFunctionsImpl"
40 #define UNIQUE false // function name does not exist in Calc
42 #define STDPAR false // all parameters are described
43 #define INTPAR true // first parameter is internal
45 #define FUNCDATA( FuncName, ParamCount, Category, Double, IntPar ) \
46 { "get" #FuncName, DATE_FUNCNAME_##FuncName, DATE_FUNCDESC_##FuncName, DATE_DEFFUNCNAME_##FuncName, ParamCount, Category, Double, IntPar }
48 const ScaFuncDataBase pFuncDataArr[] =
50 FUNCDATA( DiffWeeks, 3, ScaCategory::DateTime, UNIQUE, INTPAR ),
51 FUNCDATA( DiffMonths, 3, ScaCategory::DateTime, UNIQUE, INTPAR ),
52 FUNCDATA( DiffYears, 3, ScaCategory::DateTime, UNIQUE, INTPAR ),
53 FUNCDATA( IsLeapYear, 1, ScaCategory::DateTime, UNIQUE, INTPAR ),
54 FUNCDATA( DaysInMonth, 1, ScaCategory::DateTime, UNIQUE, INTPAR ),
55 FUNCDATA( DaysInYear, 1, ScaCategory::DateTime, UNIQUE, INTPAR ),
56 FUNCDATA( WeeksInYear, 1, ScaCategory::DateTime, UNIQUE, INTPAR ),
57 FUNCDATA( Rot13, 1, ScaCategory::Text, UNIQUE, STDPAR )
60 #undef FUNCDATA
62 ScaFuncData::ScaFuncData(const ScaFuncDataBase& rBaseData) :
63 aIntName( OUString::createFromAscii( rBaseData.pIntName ) ),
64 pUINameID( rBaseData.pUINameID ),
65 pDescrID( rBaseData.pDescrID ),
66 nParamCount( rBaseData.nParamCount ),
67 eCat( rBaseData.eCat ),
68 bDouble( rBaseData.bDouble ),
69 bWithOpt( rBaseData.bWithOpt )
71 aCompList.push_back(OUString::createFromAscii(rBaseData.pCompListID[0]));
72 aCompList.push_back(OUString::createFromAscii(rBaseData.pCompListID[1]));
75 sal_uInt16 ScaFuncData::GetStrIndex( sal_uInt16 nParam ) const
77 if( !bWithOpt )
78 nParam++;
79 return (nParam > nParamCount) ? (nParamCount * 2) : (nParam * 2);
82 static void InitScaFuncDataList(ScaFuncDataList& rList)
84 for (const auto & nIndex : pFuncDataArr)
85 rList.push_back(ScaFuncData(nIndex));
88 // entry points for service registration / instantiation
90 extern "C" SAL_DLLPUBLIC_EXPORT css::uno::XInterface*
91 scaddins_ScaDateAddIn_get_implementation(
92 css::uno::XComponentContext* , css::uno::Sequence<css::uno::Any> const&)
94 return cppu::acquire(static_cast<cppu::OWeakObject*>(new ScaDateAddIn()));
98 // "normal" service implementation
99 ScaDateAddIn::ScaDateAddIn()
103 static const char* pLang[] = { "de", "en" };
104 static const char* pCoun[] = { "DE", "US" };
105 const sal_uInt32 nNumOfLoc = SAL_N_ELEMENTS( pLang );
107 void ScaDateAddIn::InitDefLocales()
109 pDefLocales.reset(new lang::Locale[ nNumOfLoc ]);
111 for( sal_uInt32 nIndex = 0; nIndex < nNumOfLoc; nIndex++ )
113 pDefLocales[ nIndex ].Language = OUString::createFromAscii( pLang[ nIndex ] );
114 pDefLocales[ nIndex ].Country = OUString::createFromAscii( pCoun[ nIndex ] );
118 const lang::Locale& ScaDateAddIn::GetLocale( sal_uInt32 nIndex )
120 if( !pDefLocales )
121 InitDefLocales();
123 return (nIndex < sizeof( pLang )) ? pDefLocales[ nIndex ] : aFuncLoc;
126 void ScaDateAddIn::InitData()
128 aResLocale = Translate::Create("sca", LanguageTag(aFuncLoc));
129 pFuncDataList.reset();
131 pFuncDataList.reset(new ScaFuncDataList);
132 InitScaFuncDataList(*pFuncDataList);
134 if( pDefLocales )
136 pDefLocales.reset();
140 OUString ScaDateAddIn::GetFuncDescrStr(const char** pResId, sal_uInt16 nStrIndex)
142 return ScaResId(pResId[nStrIndex - 1]);
145 // XServiceName
146 OUString SAL_CALL ScaDateAddIn::getServiceName()
148 // name of specific AddIn service
149 return MY_SERVICE;
152 // XServiceInfo
153 OUString SAL_CALL ScaDateAddIn::getImplementationName()
155 return MY_IMPLNAME;
158 sal_Bool SAL_CALL ScaDateAddIn::supportsService( const OUString& aServiceName )
160 return cppu::supportsService(this, aServiceName);
163 uno::Sequence< OUString > SAL_CALL ScaDateAddIn::getSupportedServiceNames()
165 return { ADDIN_SERVICE, MY_SERVICE };
168 // XLocalizable
169 void SAL_CALL ScaDateAddIn::setLocale( const lang::Locale& eLocale )
171 aFuncLoc = eLocale;
172 InitData(); // change of locale invalidates resources!
175 lang::Locale SAL_CALL ScaDateAddIn::getLocale()
177 return aFuncLoc;
180 OUString SAL_CALL ScaDateAddIn::getProgrammaticFuntionName( const OUString& )
182 // not used by calc
183 // (but should be implemented for other uses of the AddIn service)
184 return OUString();
187 OUString SAL_CALL ScaDateAddIn::getDisplayFunctionName( const OUString& aProgrammaticName )
189 OUString aRet;
191 auto fDataIt = std::find_if(pFuncDataList->begin(), pFuncDataList->end(),
192 FindScaFuncData( aProgrammaticName ) );
193 if( fDataIt != pFuncDataList->end() )
195 aRet = ScaResId(fDataIt->GetUINameID());
196 if( fDataIt->IsDouble() )
197 aRet += "_ADD";
199 else
201 aRet = "UNKNOWNFUNC_" + aProgrammaticName;
204 return aRet;
207 OUString SAL_CALL ScaDateAddIn::getFunctionDescription( const OUString& aProgrammaticName )
209 OUString aRet;
211 auto fDataIt = std::find_if(pFuncDataList->begin(), pFuncDataList->end(),
212 FindScaFuncData( aProgrammaticName ) );
213 if( fDataIt != pFuncDataList->end() )
214 aRet = GetFuncDescrStr( fDataIt->GetDescrID(), 1 );
216 return aRet;
219 OUString SAL_CALL ScaDateAddIn::getDisplayArgumentName(
220 const OUString& aProgrammaticName, sal_Int32 nArgument )
222 OUString aRet;
224 auto fDataIt = std::find_if(pFuncDataList->begin(), pFuncDataList->end(),
225 FindScaFuncData( aProgrammaticName ) );
226 if( fDataIt != pFuncDataList->end() && (nArgument <= 0xFFFF) )
228 sal_uInt16 nStr = fDataIt->GetStrIndex( static_cast< sal_uInt16 >( nArgument ) );
229 if( nStr )
230 aRet = GetFuncDescrStr( fDataIt->GetDescrID(), nStr );
231 else
232 aRet = "internal";
235 return aRet;
238 OUString SAL_CALL ScaDateAddIn::getArgumentDescription(
239 const OUString& aProgrammaticName, sal_Int32 nArgument )
241 OUString aRet;
243 auto fDataIt = std::find_if(pFuncDataList->begin(), pFuncDataList->end(),
244 FindScaFuncData( aProgrammaticName ) );
245 if( fDataIt != pFuncDataList->end() && (nArgument <= 0xFFFF) )
247 sal_uInt16 nStr = fDataIt->GetStrIndex( static_cast< sal_uInt16 >( nArgument ) );
248 if( nStr )
249 aRet = GetFuncDescrStr( fDataIt->GetDescrID(), nStr + 1 );
250 else
251 aRet = "for internal use only";
254 return aRet;
257 OUString SAL_CALL ScaDateAddIn::getProgrammaticCategoryName(
258 const OUString& aProgrammaticName )
260 OUString aRet;
262 auto fDataIt = std::find_if(pFuncDataList->begin(), pFuncDataList->end(),
263 FindScaFuncData( aProgrammaticName ) );
264 if( fDataIt != pFuncDataList->end() )
266 switch( fDataIt->GetCategory() )
268 case ScaCategory::DateTime: aRet = "Date&Time"; break;
269 case ScaCategory::Text: aRet = "Text"; break;
270 case ScaCategory::Finance: aRet = "Financial"; break;
271 case ScaCategory::Inf: aRet = "Information"; break;
272 case ScaCategory::Math: aRet = "Mathematical"; break;
273 case ScaCategory::Tech: aRet = "Technical"; break;
277 if( aRet.isEmpty() )
278 aRet = "Add-In";
279 return aRet;
282 OUString SAL_CALL ScaDateAddIn::getDisplayCategoryName(
283 const OUString& aProgrammaticName )
285 return getProgrammaticCategoryName( aProgrammaticName );
288 // XCompatibilityNames
289 uno::Sequence< sheet::LocalizedName > SAL_CALL ScaDateAddIn::getCompatibilityNames(
290 const OUString& aProgrammaticName )
292 auto fDataIt = std::find_if(pFuncDataList->begin(), pFuncDataList->end(),
293 FindScaFuncData( aProgrammaticName ) );
294 if( fDataIt == pFuncDataList->end() )
295 return uno::Sequence< sheet::LocalizedName >( 0 );
297 const std::vector<OUString>& rStrList = fDataIt->GetCompNameList();
298 sal_uInt32 nCount = rStrList.size();
300 uno::Sequence< sheet::LocalizedName > aRet( nCount );
301 sheet::LocalizedName* pArray = aRet.getArray();
303 for( sal_uInt32 nIndex = 0; nIndex < nCount; nIndex++ )
304 pArray[ nIndex ] = sheet::LocalizedName( GetLocale( nIndex ), rStrList.at( nIndex ) );
306 return aRet;
309 namespace {
311 // auxiliary functions
312 bool IsLeapYear( sal_uInt16 nYear )
314 return ((((nYear % 4) == 0) && ((nYear % 100) != 0)) || ((nYear % 400) == 0));
317 sal_uInt16 DaysInMonth( sal_uInt16 nMonth, sal_uInt16 nYear )
319 static const sal_uInt16 aDaysInMonth[12] = { 31, 28, 31, 30, 31, 30,
320 31, 31, 30, 31, 30, 31 };
322 if ( nMonth != 2 )
323 return aDaysInMonth[nMonth-1];
324 else
326 if ( IsLeapYear(nYear) )
327 return aDaysInMonth[nMonth-1] + 1;
328 else
329 return aDaysInMonth[nMonth-1];
334 * Convert a date to a count of days starting from 01/01/0001
336 * The internal representation of a Date used in this Addin
337 * is the number of days between 01/01/0001 and the date
338 * this function converts a Day , Month, Year representation
339 * to this internal Date value.
342 sal_Int32 DateToDays( sal_uInt16 nDay, sal_uInt16 nMonth, sal_uInt16 nYear )
344 sal_Int32 nDays = (static_cast<sal_Int32>(nYear)-1) * 365;
345 nDays += ((nYear-1) / 4) - ((nYear-1) / 100) + ((nYear-1) / 400);
347 for( sal_uInt16 i = 1; i < nMonth; i++ )
348 nDays += DaysInMonth(i,nYear);
349 nDays += nDay;
351 return nDays;
355 * Convert a count of days starting from 01/01/0001 to a date
357 * The internal representation of a Date used in this Addin
358 * is the number of days between 01/01/0001 and the date
359 * this function converts this internal Date value
360 * to a Day , Month, Year representation of a Date.
362 * @throws lang::IllegalArgumentException
365 void DaysToDate( sal_Int32 nDays,
366 sal_uInt16& rDay, sal_uInt16& rMonth, sal_uInt16& rYear )
368 if( nDays < 0 )
369 throw lang::IllegalArgumentException();
371 sal_Int32 nTempDays;
372 sal_Int32 i = 0;
373 bool bCalc;
377 nTempDays = nDays;
378 rYear = static_cast<sal_uInt16>((nTempDays / 365) - i);
379 nTempDays -= (static_cast<sal_Int32>(rYear) -1) * 365;
380 nTempDays -= (( rYear -1) / 4) - (( rYear -1) / 100) + ((rYear -1) / 400);
381 bCalc = false;
382 if ( nTempDays < 1 )
384 i++;
385 bCalc = true;
387 else
389 if ( nTempDays > 365 )
391 if ( (nTempDays != 366) || !IsLeapYear( rYear ) )
393 i--;
394 bCalc = true;
399 while ( bCalc );
401 rMonth = 1;
402 while ( nTempDays > DaysInMonth( rMonth, rYear ) )
404 nTempDays -= DaysInMonth( rMonth, rYear );
405 rMonth++;
407 rDay = static_cast<sal_uInt16>(nTempDays);
411 * Get the null date used by the spreadsheet document
413 * The internal representation of a Date used in this Addin
414 * is the number of days between 01/01/0001 and the date
415 * this function returns this internal Date value for the document null date
417 * @throws uno::RuntimeException
419 sal_Int32 GetNullDate( const uno::Reference< beans::XPropertySet >& xOptions )
421 if (xOptions.is())
425 uno::Any aAny = xOptions->getPropertyValue( "NullDate" );
426 util::Date aDate;
427 if ( aAny >>= aDate )
428 return DateToDays( aDate.Day, aDate.Month, aDate.Year );
430 catch (uno::Exception&)
435 // no null date available -> no calculations possible
436 throw uno::RuntimeException();
440 // XDateFunctions
443 * Get week difference between 2 dates
445 * new Weeks(date1,date2,mode) function for StarCalc
447 * Two modes of operation are provided.
448 * The first is just a simple division by 7 calculation.
450 * The second calculates the difference by week of year.
452 * The International Standard IS-8601 has decreed that Monday
453 * shall be the first day of the week.
455 * A week that lies partly in one year and partly in another
456 * is assigned a number in the year in which most of its days lie.
458 * That means that week 1 of any year is the week that contains the 4. January
460 * The internal representation of a Date used in the Addin is the number of days based on 01/01/0001
462 * A WeekDay can be then calculated by subtracting 1 and calculating the rest of
463 * a division by 7, which gives a 0 - 6 value for Monday - Sunday
465 * Using the 4. January rule explained above the formula
467 * nWeek1= ( nDays1 - nJan4 + ( (nJan4-1) % 7 ) ) / 7 + 1;
469 * calculates a number between 0-53 for each day which is in the same year as nJan4
470 * where 0 means that this week belonged to the year before.
472 * If a day in the same or another year is used in this formula this calculates
473 * a calendar week offset from a given 4. January
475 * nWeek2 = ( nDays2 - nJan4 + ( (nJan4-1) % 7 ) ) / 7 + 1;
477 * The 4.January of first Date Argument can thus be used to calculate
478 * the week difference by calendar weeks which is then nWeek = nWeek2 - nWeek1
480 * which can be optimized to
482 * nWeek = ( (nDays2-nJan4+((nJan4-1)%7))/7 ) - ( (nDays1-nJan4+((nJan4-1)%7))/7 )
484 * Note: All calculations are operating on the long integer data type
485 * % is the modulo operator in C which calculates the rest of an Integer division
488 * mode 0 is the interval between the dates in month, that is days / 7
490 * mode 1 is the difference by week of year
494 sal_Int32 SAL_CALL ScaDateAddIn::getDiffWeeks(
495 const uno::Reference< beans::XPropertySet >& xOptions,
496 sal_Int32 nStartDate, sal_Int32 nEndDate,
497 sal_Int32 nMode )
499 if (nMode != 0 && nMode != 1)
500 throw lang::IllegalArgumentException();
502 sal_Int32 nNullDate = GetNullDate( xOptions );
504 sal_Int32 nDays1 = nStartDate + nNullDate;
505 sal_Int32 nDays2 = nEndDate + nNullDate;
507 sal_Int32 nRet;
509 if ( nMode == 1 )
511 sal_uInt16 nDay,nMonth,nYear;
512 DaysToDate( nDays1, nDay, nMonth, nYear );
513 sal_Int32 nJan4 = DateToDays( 4, 1, nYear );
515 nRet = ( (nDays2-nJan4+((nJan4-1)%7))/7 ) - ( (nDays1-nJan4+((nJan4-1)%7))/7 );
517 else
519 nRet = (nDays2 - nDays1) / 7;
521 return nRet;
525 * Get month difference between 2 dates
526 * =Month(start, end, mode) Function for StarCalc
528 * two modes are provided
530 * mode 0 is the interval between the dates in month
532 * mode 1 is the difference in calendar month
534 sal_Int32 SAL_CALL ScaDateAddIn::getDiffMonths(
535 const uno::Reference< beans::XPropertySet >& xOptions,
536 sal_Int32 nStartDate, sal_Int32 nEndDate,
537 sal_Int32 nMode )
539 if (nMode != 0 && nMode != 1)
540 throw lang::IllegalArgumentException();
542 sal_Int32 nNullDate = GetNullDate( xOptions );
544 sal_Int32 nDays1 = nStartDate + nNullDate;
545 sal_Int32 nDays2 = nEndDate + nNullDate;
547 sal_uInt16 nDay1,nMonth1,nYear1;
548 sal_uInt16 nDay2,nMonth2,nYear2;
549 DaysToDate(nDays1,nDay1,nMonth1,nYear1);
550 DaysToDate(nDays2,nDay2,nMonth2,nYear2);
552 sal_Int32 nRet = nMonth2 - nMonth1 + (nYear2 - nYear1) * 12;
553 if ( nMode == 1 || nDays1 == nDays2 ) return nRet;
555 if ( nDays1 < nDays2 )
557 if ( nDay1 > nDay2 )
559 nRet -= 1;
562 else
564 if ( nDay1 < nDay2 )
566 nRet += 1;
570 return nRet;
574 * Get Year difference between 2 dates
576 * two modes are provided
578 * mode 0 is the interval between the dates in years
580 * mode 1 is the difference in calendar years
582 sal_Int32 SAL_CALL ScaDateAddIn::getDiffYears(
583 const uno::Reference< beans::XPropertySet >& xOptions,
584 sal_Int32 nStartDate, sal_Int32 nEndDate,
585 sal_Int32 nMode )
587 if (nMode != 0 && nMode != 1)
588 throw lang::IllegalArgumentException();
590 if ( nMode != 1 )
591 return getDiffMonths( xOptions, nStartDate, nEndDate, nMode ) / 12;
593 sal_Int32 nNullDate = GetNullDate( xOptions );
595 sal_Int32 nDays1 = nStartDate + nNullDate;
596 sal_Int32 nDays2 = nEndDate + nNullDate;
598 sal_uInt16 nDay1,nMonth1,nYear1;
599 sal_uInt16 nDay2,nMonth2,nYear2;
600 DaysToDate(nDays1,nDay1,nMonth1,nYear1);
601 DaysToDate(nDays2,nDay2,nMonth2,nYear2);
603 return nYear2 - nYear1;
607 * Check if a Date is in a leap year in the Gregorian calendar
609 sal_Int32 SAL_CALL ScaDateAddIn::getIsLeapYear(
610 const uno::Reference< beans::XPropertySet >& xOptions,
611 sal_Int32 nDate )
613 sal_Int32 nNullDate = GetNullDate( xOptions );
614 sal_Int32 nDays = nDate + nNullDate;
616 sal_uInt16 nDay, nMonth, nYear;
617 DaysToDate(nDays,nDay,nMonth,nYear);
619 return static_cast<sal_Int32>(IsLeapYear(nYear));
623 * Get the Number of Days in the month for a date
625 sal_Int32 SAL_CALL ScaDateAddIn::getDaysInMonth(
626 const uno::Reference<beans::XPropertySet>& xOptions,
627 sal_Int32 nDate )
629 sal_Int32 nNullDate = GetNullDate( xOptions );
630 sal_Int32 nDays = nDate + nNullDate;
632 sal_uInt16 nDay, nMonth, nYear;
633 DaysToDate(nDays,nDay,nMonth,nYear);
635 return DaysInMonth( nMonth, nYear );
639 * Get number of days in the year of a date specified
641 sal_Int32 SAL_CALL ScaDateAddIn::getDaysInYear(
642 const uno::Reference< beans::XPropertySet >& xOptions,
643 sal_Int32 nDate )
645 sal_Int32 nNullDate = GetNullDate( xOptions );
646 sal_Int32 nDays = nDate + nNullDate;
648 sal_uInt16 nDay, nMonth, nYear;
649 DaysToDate(nDays,nDay,nMonth,nYear);
651 return ( IsLeapYear(nYear) ? 366 : 365 );
655 * Get number of weeks in the year for a date
657 * Most years have 52 weeks, but years that start on a Thursday
658 * and leap years that start on a Wednesday have 53 weeks
660 * The International Standard IS-8601 has decreed that Monday
661 * shall be the first day of the week.
663 * A WeekDay can be calculated by subtracting 1 and calculating the rest of
664 * a division by 7 from the internal date representation
665 * which gives a 0 - 6 value for Monday - Sunday
667 * @see #IsLeapYear #WeekNumber
669 sal_Int32 SAL_CALL ScaDateAddIn::getWeeksInYear(
670 const uno::Reference< beans::XPropertySet >& xOptions,
671 sal_Int32 nDate )
673 sal_Int32 nNullDate = GetNullDate( xOptions );
674 sal_Int32 nDays = nDate + nNullDate;
676 sal_uInt16 nDay, nMonth, nYear;
677 DaysToDate(nDays,nDay,nMonth,nYear);
679 sal_Int32 nJan1WeekDay = ( DateToDays(1,1,nYear) - 1) % 7;
681 sal_Int32 nRet;
682 if ( nJan1WeekDay == 3 ) /* Thursday */
683 nRet = 53;
684 else if ( nJan1WeekDay == 2 ) /* Wednesday */
685 nRet = ( IsLeapYear(nYear) ? 53 : 52 );
686 else
687 nRet = 52;
689 return nRet;
693 * Encrypt or decrypt a string using ROT13 algorithm
695 * This function rotates each character by 13 in the alphabet.
696 * Only the characters 'a' ... 'z' and 'A' ... 'Z' are modified.
698 OUString SAL_CALL ScaDateAddIn::getRot13( const OUString& aSrcString )
700 OUStringBuffer aBuffer( aSrcString );
701 for( sal_Int32 nIndex = 0; nIndex < aBuffer.getLength(); nIndex++ )
703 sal_Unicode cChar = aBuffer[nIndex];
704 if( (cChar >= 'a') && (cChar <= 'z'))
706 cChar += 13;
707 if (cChar > 'z')
708 cChar -= 26;
710 else if( (cChar >= 'A') && (cChar <= 'Z') )
712 cChar += 13;
713 if (cChar > 'Z')
714 cChar -= 26;
716 aBuffer[nIndex] = cChar;
718 return aBuffer.makeStringAndClear();
721 OUString ScaDateAddIn::ScaResId(const char* pId)
723 return Translate::get(pId, aResLocale);
726 /* vim:set shiftwidth=4 softtabstop=4 expandtab: */