xref: /trunk/main/basic/source/sbx/sbxscan.cxx (revision 29d7882dfd44bedddaf81c3b4e2999e0ea40948d)
1e1f63238SAndrew Rist /**************************************************************
2cdf0e10cSrcweir  *
3e1f63238SAndrew Rist  * Licensed to the Apache Software Foundation (ASF) under one
4e1f63238SAndrew Rist  * or more contributor license agreements.  See the NOTICE file
5e1f63238SAndrew Rist  * distributed with this work for additional information
6e1f63238SAndrew Rist  * regarding copyright ownership.  The ASF licenses this file
7e1f63238SAndrew Rist  * to you under the Apache License, Version 2.0 (the
8e1f63238SAndrew Rist  * "License"); you may not use this file except in compliance
9e1f63238SAndrew Rist  * with the License.  You may obtain a copy of the License at
10cdf0e10cSrcweir  *
11e1f63238SAndrew Rist  *   http://www.apache.org/licenses/LICENSE-2.0
12cdf0e10cSrcweir  *
13e1f63238SAndrew Rist  * Unless required by applicable law or agreed to in writing,
14e1f63238SAndrew Rist  * software distributed under the License is distributed on an
15e1f63238SAndrew Rist  * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
16e1f63238SAndrew Rist  * KIND, either express or implied.  See the License for the
17e1f63238SAndrew Rist  * specific language governing permissions and limitations
18e1f63238SAndrew Rist  * under the License.
19cdf0e10cSrcweir  *
20e1f63238SAndrew Rist  *************************************************************/
21e1f63238SAndrew Rist 
22cdf0e10cSrcweir // MARKER(update_precomp.py): autogen include statement, do not remove
23cdf0e10cSrcweir #include "precompiled_basic.hxx"
24cdf0e10cSrcweir #include <tools/errcode.hxx>
25cdf0e10cSrcweir #include <basic/sbx.hxx>
26cdf0e10cSrcweir #include "sbxconv.hxx"
27cdf0e10cSrcweir 
28cdf0e10cSrcweir #include "unotools/syslocale.hxx"
29cdf0e10cSrcweir 
30cdf0e10cSrcweir #if defined ( UNX )
31cdf0e10cSrcweir #include <stdlib.h>
32cdf0e10cSrcweir #endif
33cdf0e10cSrcweir 
34cdf0e10cSrcweir #ifndef _APP_HXX //autogen
35cdf0e10cSrcweir #include <vcl/svapp.hxx>
36cdf0e10cSrcweir #endif
37cdf0e10cSrcweir #include <math.h>
38cdf0e10cSrcweir #include <string.h>
39cdf0e10cSrcweir #include <ctype.h>
40cdf0e10cSrcweir 
41cdf0e10cSrcweir #include "sbxres.hxx"
42cdf0e10cSrcweir #include <basic/sbxbase.hxx>
43cdf0e10cSrcweir #include <basic/sbxform.hxx>
44cdf0e10cSrcweir #include <svtools/svtools.hrc>
45cdf0e10cSrcweir 
46cdf0e10cSrcweir #include "basrid.hxx"
47cdf0e10cSrcweir #include "runtime.hxx"
48cdf0e10cSrcweir 
49cdf0e10cSrcweir #include <svl/zforlist.hxx>
50cdf0e10cSrcweir #include <comphelper/processfactory.hxx>
51cdf0e10cSrcweir 
52cdf0e10cSrcweir 
ImpGetIntntlSep(sal_Unicode & rcDecimalSep,sal_Unicode & rcThousandSep)53cdf0e10cSrcweir void ImpGetIntntlSep( sal_Unicode& rcDecimalSep, sal_Unicode& rcThousandSep )
54cdf0e10cSrcweir {
55cdf0e10cSrcweir     SvtSysLocale aSysLocale;
56cdf0e10cSrcweir     const LocaleDataWrapper& rData = aSysLocale.GetLocaleData();
57cdf0e10cSrcweir     rcDecimalSep = rData.getNumDecimalSep().GetBuffer()[0];
58cdf0e10cSrcweir     rcThousandSep = rData.getNumThousandSep().GetBuffer()[0];
59cdf0e10cSrcweir }
60cdf0e10cSrcweir 
61cdf0e10cSrcweir // Scannen eines Strings nach BASIC-Konventionen
62*29d7882dSmseidel // Dies entspricht den üblichen Konventionen, nur dass der Exponent
63cdf0e10cSrcweir // auch ein D sein darf, was den Datentyp auf SbxDOUBLE festlegt.
64*29d7882dSmseidel // Die Routine versucht, den Datentyp so klein wie möglich zu gestalten.
65cdf0e10cSrcweir // Das ganze gibt auch noch einen Konversionsfehler, wenn der Datentyp
66cdf0e10cSrcweir // Fixed ist und das ganze nicht hineinpasst!
67cdf0e10cSrcweir 
ImpScan(const::rtl::OUString & rWSrc,double & nVal,SbxDataType & rType,sal_uInt16 * pLen,sal_Bool bAllowIntntl,sal_Bool bOnlyIntntl)68cdf0e10cSrcweir SbxError ImpScan( const ::rtl::OUString& rWSrc, double& nVal, SbxDataType& rType,
69cdf0e10cSrcweir                   sal_uInt16* pLen, sal_Bool bAllowIntntl, sal_Bool bOnlyIntntl )
70cdf0e10cSrcweir {
71cdf0e10cSrcweir     ::rtl::OString aBStr( ::rtl::OUStringToOString( rWSrc, RTL_TEXTENCODING_ASCII_US ) );
72cdf0e10cSrcweir 
73cdf0e10cSrcweir     // Bei International Komma besorgen
74cdf0e10cSrcweir     char cIntntlComma, cIntntl1000;
75cdf0e10cSrcweir     char cNonIntntlComma = '.';
76cdf0e10cSrcweir 
77cdf0e10cSrcweir     sal_Unicode cDecimalSep, cThousandSep = 0;
78cdf0e10cSrcweir     if( bAllowIntntl || bOnlyIntntl )
79cdf0e10cSrcweir     {
80cdf0e10cSrcweir         ImpGetIntntlSep( cDecimalSep, cThousandSep );
81cdf0e10cSrcweir         cIntntlComma = (char)cDecimalSep;
82cdf0e10cSrcweir         cIntntl1000 = (char)cThousandSep;
83cdf0e10cSrcweir     }
84cdf0e10cSrcweir     // Sonst einfach auch auf . setzen
85cdf0e10cSrcweir     else
86cdf0e10cSrcweir     {
87cdf0e10cSrcweir         cIntntlComma = cNonIntntlComma;
88*29d7882dSmseidel         cIntntl1000 = cNonIntntlComma;  // Unschädlich machen
89cdf0e10cSrcweir     }
90*29d7882dSmseidel     // Nur International -> IntnlComma übernehmen
91cdf0e10cSrcweir     if( bOnlyIntntl )
92cdf0e10cSrcweir     {
93cdf0e10cSrcweir         cNonIntntlComma = cIntntlComma;
94cdf0e10cSrcweir         cIntntl1000 = (char)cThousandSep;
95cdf0e10cSrcweir     }
96cdf0e10cSrcweir 
97cdf0e10cSrcweir     const char* pStart = aBStr.getStr();
98cdf0e10cSrcweir     const char* p = pStart;
99cdf0e10cSrcweir     char buf[ 80 ], *q = buf;
100cdf0e10cSrcweir     sal_Bool bRes = sal_True;
101cdf0e10cSrcweir     sal_Bool bMinus = sal_False;
102cdf0e10cSrcweir     nVal = 0;
103cdf0e10cSrcweir     SbxDataType eScanType = SbxSINGLE;
104cdf0e10cSrcweir     // Whitespace wech
105cdf0e10cSrcweir     while( *p &&( *p == ' ' || *p == '\t' ) ) p++;
106cdf0e10cSrcweir     // Zahl? Dann einlesen und konvertieren.
107cdf0e10cSrcweir     if( *p == '-' )
108cdf0e10cSrcweir         p++, bMinus = sal_True;
109cdf0e10cSrcweir     if( isdigit( *p ) ||( (*p == cNonIntntlComma || *p == cIntntlComma ||
110cdf0e10cSrcweir             *p == cIntntl1000) && isdigit( *(p+1 ) ) ) )
111cdf0e10cSrcweir     {
112cdf0e10cSrcweir         short exp = 0;      // >0: Exponentteil
113cdf0e10cSrcweir         short comma = 0;    // >0: Nachkomma
114cdf0e10cSrcweir         short ndig = 0;     // Anzahl Ziffern
115cdf0e10cSrcweir         short ncdig = 0;    // Anzahl Ziffern nach Komma
116cdf0e10cSrcweir         ByteString aSearchStr( "0123456789DEde" );
117cdf0e10cSrcweir         // Kommas ergaenzen
118cdf0e10cSrcweir         aSearchStr += cNonIntntlComma;
119cdf0e10cSrcweir         if( cIntntlComma != cNonIntntlComma )
120cdf0e10cSrcweir             aSearchStr += cIntntlComma;
121cdf0e10cSrcweir         if( bOnlyIntntl )
122cdf0e10cSrcweir             aSearchStr += cIntntl1000;
123cdf0e10cSrcweir         const char* pSearchStr = aSearchStr.GetBuffer();
124cdf0e10cSrcweir         while( strchr( pSearchStr, *p ) && *p )
125cdf0e10cSrcweir         {
126*29d7882dSmseidel             // 1000er-Trenner überlesen
127cdf0e10cSrcweir             if( bOnlyIntntl && *p == cIntntl1000 )
128cdf0e10cSrcweir             {
129cdf0e10cSrcweir                 p++;
130cdf0e10cSrcweir                 continue;
131cdf0e10cSrcweir             }
132cdf0e10cSrcweir 
133cdf0e10cSrcweir             // Komma oder Exponent?
134cdf0e10cSrcweir             if( *p == cNonIntntlComma || *p == cIntntlComma )
135cdf0e10cSrcweir             {
136*29d7882dSmseidel                 // Immer '.' einfügen, damit atof funktioniert
137cdf0e10cSrcweir                 p++;
138cdf0e10cSrcweir                 if( ++comma > 1 )
139cdf0e10cSrcweir                     continue;
140cdf0e10cSrcweir                 else
141cdf0e10cSrcweir                     *q++ = '.';
142cdf0e10cSrcweir             }
143cdf0e10cSrcweir             else if( strchr( "DdEe", *p ) )
144cdf0e10cSrcweir             {
145cdf0e10cSrcweir                 if( ++exp > 1 )
146cdf0e10cSrcweir                 {
147cdf0e10cSrcweir                     p++; continue;
148cdf0e10cSrcweir                 }
149cdf0e10cSrcweir                 if( toupper( *p ) == 'D' )
150cdf0e10cSrcweir                     eScanType = SbxDOUBLE;
151cdf0e10cSrcweir                 *q++ = 'E'; p++;
152cdf0e10cSrcweir                 // Vorzeichen hinter Exponent?
153cdf0e10cSrcweir                 if( *p == '+' )
154cdf0e10cSrcweir                     p++;
155cdf0e10cSrcweir                 else
156cdf0e10cSrcweir                 if( *p == '-' )
157cdf0e10cSrcweir                     *q++ = *p++;
158cdf0e10cSrcweir             }
159cdf0e10cSrcweir             else
160cdf0e10cSrcweir             {
161cdf0e10cSrcweir                 *q++ = *p++;
162cdf0e10cSrcweir                 if( comma && !exp ) ncdig++;
163cdf0e10cSrcweir             }
164cdf0e10cSrcweir             if( !exp ) ndig++;
165cdf0e10cSrcweir         }
166cdf0e10cSrcweir         *q = 0;
167cdf0e10cSrcweir         // Komma, Exponent mehrfach vorhanden?
168cdf0e10cSrcweir         if( comma > 1 || exp > 1 )
169cdf0e10cSrcweir             bRes = sal_False;
170cdf0e10cSrcweir         // Kann auf Integer gefaltet werden?
171cdf0e10cSrcweir         if( !comma && !exp )
172cdf0e10cSrcweir         {
173cdf0e10cSrcweir             if( nVal >= SbxMININT && nVal <= SbxMAXINT )
174cdf0e10cSrcweir                 eScanType = SbxINTEGER;
175cdf0e10cSrcweir             else if( nVal >= SbxMINLNG && nVal <= SbxMAXLNG )
176cdf0e10cSrcweir                 eScanType = SbxLONG;
177cdf0e10cSrcweir         }
178cdf0e10cSrcweir 
179cdf0e10cSrcweir         nVal = atof( buf );
180cdf0e10cSrcweir         ndig = ndig - comma;
181*29d7882dSmseidel         // zu viele Zahlen für SINGLE?
182cdf0e10cSrcweir         if( ndig > 15 || ncdig > 6 )
183cdf0e10cSrcweir             eScanType = SbxDOUBLE;
184cdf0e10cSrcweir 
185cdf0e10cSrcweir         // Typkennung?
186cdf0e10cSrcweir         if( strchr( "%!&#", *p ) && *p ) p++;
187cdf0e10cSrcweir     }
188cdf0e10cSrcweir     // Hex/Oktalzahl? Einlesen und konvertieren:
189cdf0e10cSrcweir     else if( *p == '&' )
190cdf0e10cSrcweir     {
191cdf0e10cSrcweir         p++;
192cdf0e10cSrcweir         eScanType = SbxLONG;
193cdf0e10cSrcweir         const char *cmp = "0123456789ABCDEF";
194cdf0e10cSrcweir         char base = 16;
195cdf0e10cSrcweir         char ndig = 8;
196cdf0e10cSrcweir         char xch  = *p++;
197cdf0e10cSrcweir         switch( toupper( xch ) )
198cdf0e10cSrcweir         {
199cdf0e10cSrcweir             case 'O': cmp = "01234567"; base = 8; ndig = 11; break;
200cdf0e10cSrcweir             case 'H': break;
201cdf0e10cSrcweir             default : bRes = sal_False;
202cdf0e10cSrcweir         }
203323c3501SDamjan Jovanovic         sal_Int32 l = 0;
204cdf0e10cSrcweir         int i;
205cdf0e10cSrcweir         while( isalnum( *p ) )
206cdf0e10cSrcweir         {
207cdf0e10cSrcweir             char ch = sal::static_int_cast< char >( toupper( *p ) );
208cdf0e10cSrcweir             p++;
209cdf0e10cSrcweir             if( strchr( cmp, ch ) ) *q++ = ch;
210cdf0e10cSrcweir             else bRes = sal_False;
211cdf0e10cSrcweir         }
212cdf0e10cSrcweir         *q = 0;
213cdf0e10cSrcweir         for( q = buf; *q; q++ )
214cdf0e10cSrcweir         {
215cdf0e10cSrcweir             i =( *q & 0xFF ) - '0';
216cdf0e10cSrcweir             if( i > 9 ) i -= 7;
217cdf0e10cSrcweir             l =( l * base ) + i;
218cdf0e10cSrcweir             if( !ndig-- )
219cdf0e10cSrcweir                 bRes = sal_False;
220cdf0e10cSrcweir         }
221cdf0e10cSrcweir         if( *p == '&' ) p++;
222cdf0e10cSrcweir         nVal = (double) l;
223cdf0e10cSrcweir         if( l >= SbxMININT && l <= SbxMAXINT )
224cdf0e10cSrcweir             eScanType = SbxINTEGER;
225cdf0e10cSrcweir     }
226cdf0e10cSrcweir     else if ( SbiRuntime::isVBAEnabled() )
227cdf0e10cSrcweir     {
228cdf0e10cSrcweir         OSL_TRACE("Reporting error converting");
229cdf0e10cSrcweir         return SbxERR_CONVERSION;
230cdf0e10cSrcweir     }
231cdf0e10cSrcweir     if( pLen )
232cdf0e10cSrcweir         *pLen = (sal_uInt16) ( p - pStart );
233cdf0e10cSrcweir     if( !bRes )
234cdf0e10cSrcweir         return SbxERR_CONVERSION;
235cdf0e10cSrcweir     if( bMinus )
236cdf0e10cSrcweir         nVal = -nVal;
237cdf0e10cSrcweir     rType = eScanType;
238cdf0e10cSrcweir     return SbxERR_OK;
239cdf0e10cSrcweir }
240cdf0e10cSrcweir 
241*29d7882dSmseidel // Schnittstelle für CDbl im Basic
ScanNumIntnl(const String & rSrc,double & nVal,sal_Bool bSingle)242cdf0e10cSrcweir SbxError SbxValue::ScanNumIntnl( const String& rSrc, double& nVal, sal_Bool bSingle )
243cdf0e10cSrcweir {
244cdf0e10cSrcweir     SbxDataType t;
245cdf0e10cSrcweir     sal_uInt16 nLen = 0;
246cdf0e10cSrcweir     SbxError nRetError = ImpScan( rSrc, nVal, t, &nLen,
247cdf0e10cSrcweir         /*bAllowIntntl*/sal_False, /*bOnlyIntntl*/sal_True );
248cdf0e10cSrcweir     // Komplett gelesen?
249cdf0e10cSrcweir     if( nRetError == SbxERR_OK && nLen != rSrc.Len() )
250cdf0e10cSrcweir         nRetError = SbxERR_CONVERSION;
251cdf0e10cSrcweir 
252cdf0e10cSrcweir     if( bSingle )
253cdf0e10cSrcweir     {
254cdf0e10cSrcweir         SbxValues aValues( nVal );
255cdf0e10cSrcweir         nVal = (double)ImpGetSingle( &aValues );    // Hier Error bei Overflow
256cdf0e10cSrcweir     }
257cdf0e10cSrcweir     return nRetError;
258cdf0e10cSrcweir }
259cdf0e10cSrcweir 
260cdf0e10cSrcweir ////////////////////////////////////////////////////////////////////////////
261cdf0e10cSrcweir 
262cdf0e10cSrcweir static double roundArray[] = {
263cdf0e10cSrcweir     5.0e+0, 0.5e+0, 0.5e-1, 0.5e-2, 0.5e-3, 0.5e-4, 0.5e-5, 0.5e-6, 0.5e-7,
264cdf0e10cSrcweir     0.5e-8, 0.5e-9, 0.5e-10,0.5e-11,0.5e-12,0.5e-13,0.5e-14,0.5e-15 };
265cdf0e10cSrcweir 
266cdf0e10cSrcweir /***************************************************************************
267cdf0e10cSrcweir |*
268cdf0e10cSrcweir |*  void myftoa( double, char *, short, short, sal_Bool, sal_Bool )
269cdf0e10cSrcweir |*
270cdf0e10cSrcweir |*  Beschreibung:       Konversion double --> ASCII
271cdf0e10cSrcweir |*  Parameter:          double              die Zahl.
272cdf0e10cSrcweir |*                      char *              der Zielpuffer
273cdf0e10cSrcweir |*                      short               Anzahl Nachkommastellen
274cdf0e10cSrcweir |*                      short               Weite des Exponenten( 0=kein E )
275cdf0e10cSrcweir |*                      sal_Bool                sal_True: mit 1000er Punkten
276cdf0e10cSrcweir |*                      sal_Bool                sal_True: formatfreie Ausgabe
277cdf0e10cSrcweir |*
278cdf0e10cSrcweir ***************************************************************************/
279cdf0e10cSrcweir 
myftoa(double nNum,char * pBuf,short nPrec,short nExpWidth,sal_Bool bPt,sal_Bool bFix,sal_Unicode cForceThousandSep=0)280cdf0e10cSrcweir static void myftoa( double nNum, char * pBuf, short nPrec, short nExpWidth,
281cdf0e10cSrcweir                     sal_Bool bPt, sal_Bool bFix, sal_Unicode cForceThousandSep = 0 )
282cdf0e10cSrcweir {
283cdf0e10cSrcweir 
284cdf0e10cSrcweir     short nExp = 0;                     // Exponent
285cdf0e10cSrcweir     short nDig = nPrec + 1;             // Anzahl Digits in Zahl
286cdf0e10cSrcweir     short nDec;                         // Anzahl Vorkommastellen
287cdf0e10cSrcweir     register int i, digit;
288cdf0e10cSrcweir 
289cdf0e10cSrcweir     // Komma besorgen
290cdf0e10cSrcweir     sal_Unicode cDecimalSep, cThousandSep;
291cdf0e10cSrcweir     ImpGetIntntlSep( cDecimalSep, cThousandSep );
292cdf0e10cSrcweir     if( cForceThousandSep )
293cdf0e10cSrcweir         cThousandSep = cForceThousandSep;
294cdf0e10cSrcweir 
295cdf0e10cSrcweir     // Exponentberechnung:
296cdf0e10cSrcweir     nExp = 0;
297cdf0e10cSrcweir     if( nNum > 0.0 )
298cdf0e10cSrcweir     {
299cdf0e10cSrcweir         while( nNum <   1.0 ) nNum *= 10.0, nExp--;
300cdf0e10cSrcweir         while( nNum >= 10.0 ) nNum /= 10.0, nExp++;
301cdf0e10cSrcweir     }
302cdf0e10cSrcweir     if( !bFix && !nExpWidth )
303cdf0e10cSrcweir         nDig = nDig + nExp;
304cdf0e10cSrcweir     else if( bFix && !nPrec )
305cdf0e10cSrcweir         nDig = nExp + 1;
306cdf0e10cSrcweir 
307cdf0e10cSrcweir     // Zahl runden:
308cdf0e10cSrcweir     if( (nNum += roundArray [( nDig > 16 ) ? 16 : nDig] ) >= 10.0 )
309cdf0e10cSrcweir     {
310cdf0e10cSrcweir         nNum = 1.0;
311cdf0e10cSrcweir         ++nExp;
312cdf0e10cSrcweir         if( !nExpWidth ) ++nDig;
313cdf0e10cSrcweir     }
314cdf0e10cSrcweir 
315cdf0e10cSrcweir     // Bestimmung der Vorkommastellen:
316cdf0e10cSrcweir     if( !nExpWidth )
317cdf0e10cSrcweir     {
318cdf0e10cSrcweir         if( nExp < 0 )
319cdf0e10cSrcweir         {
320cdf0e10cSrcweir             // #41691: Auch bei bFix eine 0 spendieren
321cdf0e10cSrcweir             *pBuf++ = '0';
322cdf0e10cSrcweir             if( nPrec ) *pBuf++ = (char)cDecimalSep;
323cdf0e10cSrcweir             i = -nExp - 1;
324cdf0e10cSrcweir             if( nDig <= 0 ) i = nPrec;
325cdf0e10cSrcweir             while( i-- )    *pBuf++ = '0';
326cdf0e10cSrcweir             nDec = 0;
327cdf0e10cSrcweir         }
328cdf0e10cSrcweir         else
329cdf0e10cSrcweir             nDec = nExp+1;
330cdf0e10cSrcweir     }
331cdf0e10cSrcweir     else
332cdf0e10cSrcweir         nDec = 1;
333cdf0e10cSrcweir 
334cdf0e10cSrcweir     // Zahl ausgeben:
335cdf0e10cSrcweir     if( nDig > 0 )
336cdf0e10cSrcweir     {
337cdf0e10cSrcweir         for( i = 0 ; ; ++i )
338cdf0e10cSrcweir         {
339cdf0e10cSrcweir             if( i < 16 )
340cdf0e10cSrcweir             {
341cdf0e10cSrcweir                 digit = (int) nNum;
342cdf0e10cSrcweir                 *pBuf++ = sal::static_int_cast< char >(digit + '0');
343cdf0e10cSrcweir                 nNum =( nNum - digit ) * 10.0;
344cdf0e10cSrcweir             } else
345cdf0e10cSrcweir                 *pBuf++ = '0';
346cdf0e10cSrcweir             if( --nDig == 0 ) break;
347cdf0e10cSrcweir             if( nDec )
348cdf0e10cSrcweir             {
349cdf0e10cSrcweir                 nDec--;
350cdf0e10cSrcweir                 if( !nDec )
351cdf0e10cSrcweir                     *pBuf++ = (char)cDecimalSep;
352cdf0e10cSrcweir                 else if( !(nDec % 3 ) && bPt )
353cdf0e10cSrcweir                     *pBuf++ = (char)cThousandSep;
354cdf0e10cSrcweir             }
355cdf0e10cSrcweir         }
356cdf0e10cSrcweir     }
357cdf0e10cSrcweir 
358cdf0e10cSrcweir     // Exponent ausgeben:
359cdf0e10cSrcweir     if( nExpWidth )
360cdf0e10cSrcweir     {
361cdf0e10cSrcweir         if( nExpWidth < 3 ) nExpWidth = 3;
362cdf0e10cSrcweir         nExpWidth -= 2;
363cdf0e10cSrcweir         *pBuf++ = 'E';
364cdf0e10cSrcweir         *pBuf++ =( nExp < 0 ) ?( (nExp = -nExp ), '-' ) : '+';
365cdf0e10cSrcweir         while( nExpWidth > 3 ) *pBuf++ = '0', nExpWidth--;
366cdf0e10cSrcweir         if( nExp >= 100 || nExpWidth == 3 )
367cdf0e10cSrcweir         {
368cdf0e10cSrcweir             *pBuf++ = sal::static_int_cast< char >(nExp/100 + '0');
369cdf0e10cSrcweir             nExp %= 100;
370cdf0e10cSrcweir         }
371cdf0e10cSrcweir         if( nExp/10 || nExpWidth >= 2 )
372cdf0e10cSrcweir             *pBuf++ = sal::static_int_cast< char >(nExp/10 + '0');
373cdf0e10cSrcweir         *pBuf++ = sal::static_int_cast< char >(nExp%10 + '0');
374cdf0e10cSrcweir     }
375cdf0e10cSrcweir     *pBuf = 0;
376cdf0e10cSrcweir }
377cdf0e10cSrcweir 
378cdf0e10cSrcweir // Die Zahl wird unformatiert mit der angegebenen Anzahl NK-Stellen
379cdf0e10cSrcweir // aufbereitet. Evtl. wird ein Minus vorangestellt.
380cdf0e10cSrcweir // Diese Routine ist public, weil sie auch von den Put-Funktionen
381cdf0e10cSrcweir // der Klasse SbxImpSTRING verwendet wird.
382cdf0e10cSrcweir 
383cdf0e10cSrcweir #ifdef _MSC_VER
384cdf0e10cSrcweir #pragma optimize( "", off )
385cdf0e10cSrcweir #pragma warning(disable: 4748) // "... because optimizations are disabled ..."
386cdf0e10cSrcweir #endif
387cdf0e10cSrcweir 
ImpCvtNum(double nNum,short nPrec,::rtl::OUString & rRes,sal_Bool bCoreString)388cdf0e10cSrcweir void ImpCvtNum( double nNum, short nPrec, ::rtl::OUString& rRes, sal_Bool bCoreString )
389cdf0e10cSrcweir {
390cdf0e10cSrcweir     char *q;
391cdf0e10cSrcweir     char cBuf[ 40 ], *p = cBuf;
392cdf0e10cSrcweir 
393cdf0e10cSrcweir     sal_Unicode cDecimalSep, cThousandSep;
394cdf0e10cSrcweir     ImpGetIntntlSep( cDecimalSep, cThousandSep );
395cdf0e10cSrcweir     if( bCoreString )
396cdf0e10cSrcweir         cDecimalSep = '.';
397cdf0e10cSrcweir 
398cdf0e10cSrcweir     if( nNum < 0.0 ) {
399cdf0e10cSrcweir         nNum = -nNum;
400cdf0e10cSrcweir         *p++ = '-';
401cdf0e10cSrcweir     }
402cdf0e10cSrcweir     double dMaxNumWithoutExp = (nPrec == 6) ? 1E6 : 1E14;
403389dff79SDamjan Jovanovic     myftoa( nNum, p, nPrec,( nNum &&( nNum < 1E-1 || nNum >= dMaxNumWithoutExp ) ) ? 4:0,
404cdf0e10cSrcweir         sal_False, sal_True, cDecimalSep );
405cdf0e10cSrcweir     // Trailing Zeroes weg:
406cdf0e10cSrcweir     for( p = cBuf; *p &&( *p != 'E' ); p++ ) {}
407cdf0e10cSrcweir     q = p; p--;
408cdf0e10cSrcweir     while( nPrec && *p == '0' ) nPrec--, p--;
409cdf0e10cSrcweir     if( *p == cDecimalSep ) p--;
410cdf0e10cSrcweir     while( *q ) *++p = *q++;
411cdf0e10cSrcweir     *++p = 0;
412cdf0e10cSrcweir     rRes = ::rtl::OUString::createFromAscii( cBuf );
413cdf0e10cSrcweir }
414cdf0e10cSrcweir 
415cdf0e10cSrcweir #ifdef _MSC_VER
416cdf0e10cSrcweir #pragma optimize( "", on )
417cdf0e10cSrcweir #endif
418cdf0e10cSrcweir 
ImpConvStringExt(::rtl::OUString & rSrc,SbxDataType eTargetType)419cdf0e10cSrcweir sal_Bool ImpConvStringExt( ::rtl::OUString& rSrc, SbxDataType eTargetType )
420cdf0e10cSrcweir {
421*29d7882dSmseidel     // Merken, ob überhaupt was geändert wurde
422cdf0e10cSrcweir     sal_Bool bChanged = sal_False;
423cdf0e10cSrcweir     ::rtl::OUString aNewString;
424cdf0e10cSrcweir 
425*29d7882dSmseidel     // Nur Spezial-Fälle behandeln, als Default tun wir nichts
426cdf0e10cSrcweir     switch( eTargetType )
427cdf0e10cSrcweir     {
428*29d7882dSmseidel         // Bei Fliesskomma International berücksichtigen
429cdf0e10cSrcweir         case SbxSINGLE:
430cdf0e10cSrcweir         case SbxDOUBLE:
431cdf0e10cSrcweir         case SbxCURRENCY:
432cdf0e10cSrcweir         {
433cdf0e10cSrcweir             ::rtl::OString aBStr( ::rtl::OUStringToOString( rSrc, RTL_TEXTENCODING_ASCII_US ) );
434cdf0e10cSrcweir 
435cdf0e10cSrcweir             // Komma besorgen
436cdf0e10cSrcweir             sal_Unicode cDecimalSep, cThousandSep;
437cdf0e10cSrcweir             ImpGetIntntlSep( cDecimalSep, cThousandSep );
438cdf0e10cSrcweir             aNewString = rSrc;
439cdf0e10cSrcweir 
440cdf0e10cSrcweir             // Ersetzen, wenn DecimalSep kein '.' (nur den ersten)
441cdf0e10cSrcweir             if( cDecimalSep != (sal_Unicode)'.' )
442cdf0e10cSrcweir             {
443cdf0e10cSrcweir                 sal_Int32 nPos = aNewString.indexOf( cDecimalSep );
444cdf0e10cSrcweir                 if( nPos != -1 )
445cdf0e10cSrcweir                 {
446cdf0e10cSrcweir                     sal_Unicode* pStr = (sal_Unicode*)aNewString.getStr();
447cdf0e10cSrcweir                     pStr[nPos] = (sal_Unicode)'.';
448cdf0e10cSrcweir                     bChanged = sal_True;
449cdf0e10cSrcweir                 }
450cdf0e10cSrcweir             }
451cdf0e10cSrcweir             break;
452cdf0e10cSrcweir         }
453cdf0e10cSrcweir 
454*29d7882dSmseidel         // Bei sal_Bool sal_True und sal_False als String prüfen
455cdf0e10cSrcweir         case SbxBOOL:
456cdf0e10cSrcweir         {
457cdf0e10cSrcweir             if( rSrc.equalsIgnoreAsciiCaseAscii( "true" ) )
458cdf0e10cSrcweir             {
459cdf0e10cSrcweir                 aNewString = ::rtl::OUString::valueOf( (sal_Int32)SbxTRUE );
460cdf0e10cSrcweir                 bChanged = sal_True;
461cdf0e10cSrcweir             }
462cdf0e10cSrcweir             else
463cdf0e10cSrcweir             if( rSrc.equalsIgnoreAsciiCaseAscii( "false" ) )
464cdf0e10cSrcweir             {
465cdf0e10cSrcweir                 aNewString = ::rtl::OUString::valueOf( (sal_Int32)SbxFALSE );
466cdf0e10cSrcweir                 bChanged = sal_True;
467cdf0e10cSrcweir             }
468cdf0e10cSrcweir             break;
469cdf0e10cSrcweir         }
470cdf0e10cSrcweir         default: break;
471cdf0e10cSrcweir     }
472*29d7882dSmseidel     // String bei Änderung übernehmen
473cdf0e10cSrcweir     if( bChanged )
474cdf0e10cSrcweir         rSrc = aNewString;
475cdf0e10cSrcweir     return bChanged;
476cdf0e10cSrcweir }
477cdf0e10cSrcweir 
478cdf0e10cSrcweir 
479cdf0e10cSrcweir // Formatierte Zahlenausgabe
480cdf0e10cSrcweir // Der Returnwert ist die Anzahl Zeichen, die aus dem
481*29d7882dSmseidel // Format verwendet wurden.
482cdf0e10cSrcweir 
483cdf0e10cSrcweir #ifdef _old_format_code_
484cdf0e10cSrcweir // lasse diesen Code vorl"aufig drin, zum 'abgucken'
485cdf0e10cSrcweir // der bisherigen Implementation
486cdf0e10cSrcweir 
printfmtnum(double nNum,XubString & rRes,const XubString & rWFmt)487cdf0e10cSrcweir static sal_uInt16 printfmtnum( double nNum, XubString& rRes, const XubString& rWFmt )
488cdf0e10cSrcweir {
489cdf0e10cSrcweir     const String& rFmt = rWFmt;
490cdf0e10cSrcweir     char    cFill  = ' ';           // Fuellzeichen
491cdf0e10cSrcweir     char    cPre   = 0;             // Startzeichen( evtl. "$" )
492cdf0e10cSrcweir     short   nExpDig= 0;             // Anzahl Exponentstellen
493cdf0e10cSrcweir     short   nPrec  = 0;             // Anzahl Nachkommastellen
494cdf0e10cSrcweir     short   nWidth = 0;             // Zahlenweite gesamnt
495cdf0e10cSrcweir     short   nLen;                   // Laenge konvertierte Zahl
496cdf0e10cSrcweir     sal_Bool    bPoint = sal_False;         // sal_True: mit 1000er Kommas
497cdf0e10cSrcweir     sal_Bool    bTrail = sal_False;         // sal_True, wenn folgendes Minus
498cdf0e10cSrcweir     sal_Bool    bSign  = sal_False;         // sal_True: immer mit Vorzeichen
499cdf0e10cSrcweir     sal_Bool    bNeg   = sal_False;         // sal_True: Zahl ist negativ
500cdf0e10cSrcweir     char    cBuf [1024];            // Zahlenpuffer
501cdf0e10cSrcweir     char  * p;
502cdf0e10cSrcweir     const char* pFmt = rFmt;
503cdf0e10cSrcweir     rRes.Erase();
504cdf0e10cSrcweir     // $$ und ** abfangen. Einfach wird als Zeichen ausgegeben.
505cdf0e10cSrcweir     if( *pFmt == '$' )
506cdf0e10cSrcweir       if( *++pFmt != '$' ) rRes += '$';
507cdf0e10cSrcweir     if( *pFmt == '*' )
508cdf0e10cSrcweir       if( *++pFmt != '*' ) rRes += '*';
509cdf0e10cSrcweir 
510cdf0e10cSrcweir     switch( *pFmt++ )
511cdf0e10cSrcweir     {
512cdf0e10cSrcweir         case 0:
513cdf0e10cSrcweir             break;
514cdf0e10cSrcweir         case '+':
515cdf0e10cSrcweir             bSign = sal_True; nWidth++; break;
516cdf0e10cSrcweir         case '*':
517cdf0e10cSrcweir             nWidth++; cFill = '*';
518cdf0e10cSrcweir             if( *pFmt == '$' ) nWidth++, pFmt++, cPre = '$';
519cdf0e10cSrcweir             break;
520cdf0e10cSrcweir         case '$':
521cdf0e10cSrcweir             nWidth++; cPre = '$'; break;
522cdf0e10cSrcweir         case '#':
523cdf0e10cSrcweir         case '.':
524cdf0e10cSrcweir         case ',':
525cdf0e10cSrcweir             pFmt--; break;
526cdf0e10cSrcweir     }
527cdf0e10cSrcweir     // Vorkomma:
528cdf0e10cSrcweir     for( ;; )
529cdf0e10cSrcweir     {
530cdf0e10cSrcweir         while( *pFmt == '#' ) pFmt++, nWidth++;
531cdf0e10cSrcweir         // 1000er Kommas?
532cdf0e10cSrcweir         if( *pFmt == ',' )
533cdf0e10cSrcweir         {
534cdf0e10cSrcweir             nWidth++; pFmt++; bPoint = sal_True;
535cdf0e10cSrcweir         } else break;
536cdf0e10cSrcweir     }
537cdf0e10cSrcweir     // Nachkomma:
538cdf0e10cSrcweir     if( *pFmt == '.' )
539cdf0e10cSrcweir     {
540cdf0e10cSrcweir         while( *++pFmt == '#' ) nPrec++;
541cdf0e10cSrcweir         nWidth += nPrec + 1;
542cdf0e10cSrcweir     }
543cdf0e10cSrcweir     // Exponent:
544cdf0e10cSrcweir     while( *pFmt == '^' )
545cdf0e10cSrcweir         pFmt++, nExpDig++, nWidth++;
546cdf0e10cSrcweir     // Folgendes Minus:
547cdf0e10cSrcweir     if( !bSign && *pFmt == '-' )
548cdf0e10cSrcweir         pFmt++, bTrail = sal_True;
549cdf0e10cSrcweir 
550cdf0e10cSrcweir     // Zahl konvertieren:
551cdf0e10cSrcweir     if( nPrec > 15 ) nPrec = 15;
552cdf0e10cSrcweir     if( nNum < 0.0 ) nNum = -nNum, bNeg = sal_True;
553cdf0e10cSrcweir     p = cBuf;
554cdf0e10cSrcweir     if( bSign ) *p++ = bNeg ? '-' : '+';
555cdf0e10cSrcweir     myftoa( nNum, p, nPrec, nExpDig, bPoint, sal_False );
556cdf0e10cSrcweir     nLen = strlen( cBuf );
557cdf0e10cSrcweir 
558*29d7882dSmseidel     // Überlauf?
559cdf0e10cSrcweir     if( cPre ) nLen++;
560cdf0e10cSrcweir     if( nLen > nWidth ) rRes += '%';
561cdf0e10cSrcweir     else {
562cdf0e10cSrcweir         nWidth -= nLen;
563cdf0e10cSrcweir         while( nWidth-- ) rRes += (xub_Unicode)cFill;
564cdf0e10cSrcweir         if( cPre ) rRes += (xub_Unicode)cPre;
565cdf0e10cSrcweir     }
566cdf0e10cSrcweir     rRes += (xub_Unicode*)&(cBuf[0]);
567cdf0e10cSrcweir     if( bTrail )
568cdf0e10cSrcweir         rRes += bNeg ? '-' : ' ';
569cdf0e10cSrcweir 
570cdf0e10cSrcweir     return (sal_uInt16) ( pFmt - (const char*) rFmt );
571cdf0e10cSrcweir }
572cdf0e10cSrcweir 
573cdf0e10cSrcweir #endif //_old_format_code_
574cdf0e10cSrcweir 
printfmtstr(const XubString & rStr,XubString & rRes,const XubString & rFmt)575cdf0e10cSrcweir static sal_uInt16 printfmtstr( const XubString& rStr, XubString& rRes, const XubString& rFmt )
576cdf0e10cSrcweir {
577cdf0e10cSrcweir     const xub_Unicode* pStr = rStr.GetBuffer();
578cdf0e10cSrcweir     const xub_Unicode* pFmtStart = rFmt.GetBuffer();
579cdf0e10cSrcweir     const xub_Unicode* pFmt = pFmtStart;
580cdf0e10cSrcweir     rRes.Erase();
581cdf0e10cSrcweir     switch( *pFmt )
582cdf0e10cSrcweir     {
583cdf0e10cSrcweir         case '!':
584cdf0e10cSrcweir                 rRes += *pStr++; pFmt++; break;
585cdf0e10cSrcweir         case '\\':
586cdf0e10cSrcweir             do
587cdf0e10cSrcweir             {
588cdf0e10cSrcweir                 rRes += *pStr ? *pStr++ : static_cast< xub_Unicode >(' ');
589cdf0e10cSrcweir                 pFmt++;
590cdf0e10cSrcweir             } while( *pFmt != '\\' );
591cdf0e10cSrcweir             rRes += *pStr ? *pStr++ : static_cast< xub_Unicode >(' ');
592cdf0e10cSrcweir             pFmt++; break;
593cdf0e10cSrcweir         case '&':
594cdf0e10cSrcweir             rRes = rStr;
595cdf0e10cSrcweir             pFmt++; break;
596cdf0e10cSrcweir         default:
597cdf0e10cSrcweir             rRes = rStr;
598cdf0e10cSrcweir             break;
599cdf0e10cSrcweir     }
600cdf0e10cSrcweir     return (sal_uInt16) ( pFmt - pFmtStart );
601cdf0e10cSrcweir }
602cdf0e10cSrcweir 
603cdf0e10cSrcweir /////////////////////////////////////////////////////////////////////////
604cdf0e10cSrcweir 
Scan(const XubString & rSrc,sal_uInt16 * pLen)605cdf0e10cSrcweir sal_Bool SbxValue::Scan( const XubString& rSrc, sal_uInt16* pLen )
606cdf0e10cSrcweir {
607cdf0e10cSrcweir     SbxError eRes = SbxERR_OK;
608cdf0e10cSrcweir     if( !CanWrite() )
609cdf0e10cSrcweir         eRes = SbxERR_PROP_READONLY;
610cdf0e10cSrcweir     else
611cdf0e10cSrcweir     {
612cdf0e10cSrcweir         double n;
613cdf0e10cSrcweir         SbxDataType t;
614cdf0e10cSrcweir         eRes = ImpScan( rSrc, n, t, pLen );
615cdf0e10cSrcweir         if( eRes == SbxERR_OK )
616cdf0e10cSrcweir         {
617cdf0e10cSrcweir             if( !IsFixed() )
618cdf0e10cSrcweir                 SetType( t );
619cdf0e10cSrcweir             PutDouble( n );
620cdf0e10cSrcweir         }
621cdf0e10cSrcweir     }
622cdf0e10cSrcweir     if( eRes )
623cdf0e10cSrcweir     {
624cdf0e10cSrcweir         SetError( eRes ); return sal_False;
625cdf0e10cSrcweir     }
626cdf0e10cSrcweir     else
627cdf0e10cSrcweir         return sal_True;
628cdf0e10cSrcweir }
629cdf0e10cSrcweir 
630cdf0e10cSrcweir 
implGetResMgr(void)631cdf0e10cSrcweir ResMgr* implGetResMgr( void )
632cdf0e10cSrcweir {
633cdf0e10cSrcweir     static ResMgr* pResMgr = NULL;
634cdf0e10cSrcweir     if( !pResMgr )
635cdf0e10cSrcweir     {
636cdf0e10cSrcweir         ::com::sun::star::lang::Locale aLocale = Application::GetSettings().GetUILocale();
637cdf0e10cSrcweir         pResMgr = ResMgr::CreateResMgr(CREATEVERSIONRESMGR_NAME(sb), aLocale );
638cdf0e10cSrcweir     }
639cdf0e10cSrcweir     return pResMgr;
640cdf0e10cSrcweir }
641cdf0e10cSrcweir 
642cdf0e10cSrcweir class SbxValueFormatResId : public ResId
643cdf0e10cSrcweir {
644cdf0e10cSrcweir public:
SbxValueFormatResId(sal_uInt16 nId)645cdf0e10cSrcweir     SbxValueFormatResId( sal_uInt16 nId )
646cdf0e10cSrcweir         : ResId( nId, *implGetResMgr() )
647cdf0e10cSrcweir     {}
648cdf0e10cSrcweir };
649cdf0e10cSrcweir 
650cdf0e10cSrcweir 
651cdf0e10cSrcweir enum VbaFormatType
652cdf0e10cSrcweir {
653cdf0e10cSrcweir     VBA_FORMAT_TYPE_OFFSET, // standard number format
654cdf0e10cSrcweir     VBA_FORMAT_TYPE_USERDEFINED, // user defined number format
655cdf0e10cSrcweir     VBA_FORMAT_TYPE_NULL
656cdf0e10cSrcweir };
657cdf0e10cSrcweir 
658cdf0e10cSrcweir struct VbaFormatInfo
659cdf0e10cSrcweir {
660cdf0e10cSrcweir     VbaFormatType meType;
661cdf0e10cSrcweir     const char* mpVbaFormat; // Format string in vba
662cdf0e10cSrcweir     NfIndexTableOffset meOffset; // SvNumberFormatter format index, if meType = VBA_FORMAT_TYPE_OFFSET
663cdf0e10cSrcweir     const char* mpOOoFormat; // if meType = VBA_FORMAT_TYPE_USERDEFINED
664cdf0e10cSrcweir };
665cdf0e10cSrcweir 
666cdf0e10cSrcweir #define VBA_FORMAT_OFFSET( pcUtf8, eOffset ) \
667cdf0e10cSrcweir     { VBA_FORMAT_TYPE_OFFSET, pcUtf8, eOffset, 0 }
668cdf0e10cSrcweir 
669cdf0e10cSrcweir #define VBA_FORMAT_USERDEFINED( pcUtf8, pcDefinedUtf8 ) \
670cdf0e10cSrcweir     { VBA_FORMAT_TYPE_USERDEFINED, pcUtf8, NF_NUMBER_STANDARD, pcDefinedUtf8 }
671cdf0e10cSrcweir 
672cdf0e10cSrcweir static VbaFormatInfo pFormatInfoTable[] =
673cdf0e10cSrcweir {
674cdf0e10cSrcweir     VBA_FORMAT_OFFSET( "Long Date", NF_DATE_SYSTEM_LONG ),
675cdf0e10cSrcweir     VBA_FORMAT_USERDEFINED( "Medium Date", "DD-MMM-YY" ),
676cdf0e10cSrcweir     VBA_FORMAT_OFFSET( "Short Date", NF_DATE_SYSTEM_SHORT ),
677cdf0e10cSrcweir     VBA_FORMAT_USERDEFINED( "Long Time", "H:MM:SS AM/PM" ),
678cdf0e10cSrcweir     VBA_FORMAT_OFFSET( "Medium Time", NF_TIME_HHMMAMPM ),
679cdf0e10cSrcweir     VBA_FORMAT_OFFSET( "Short Time", NF_TIME_HHMM ),
680cdf0e10cSrcweir     VBA_FORMAT_OFFSET( "ddddd", NF_DATE_SYSTEM_SHORT ),
681cdf0e10cSrcweir     VBA_FORMAT_OFFSET( "dddddd", NF_DATE_SYSTEM_LONG ),
682cdf0e10cSrcweir     VBA_FORMAT_USERDEFINED( "ttttt", "H:MM:SS AM/PM" ),
683cdf0e10cSrcweir     VBA_FORMAT_OFFSET( "ww", NF_DATE_WW ),
684cdf0e10cSrcweir     { VBA_FORMAT_TYPE_NULL, 0, NF_INDEX_TABLE_ENTRIES, 0 }
685cdf0e10cSrcweir };
686cdf0e10cSrcweir 
getFormatInfo(const String & rFmt)687cdf0e10cSrcweir VbaFormatInfo* getFormatInfo( const String& rFmt )
688cdf0e10cSrcweir {
689cdf0e10cSrcweir     VbaFormatInfo* pInfo = NULL;
690cdf0e10cSrcweir     sal_Int16 i = 0;
691cdf0e10cSrcweir     while( (pInfo = pFormatInfoTable + i )->mpVbaFormat != NULL )
692cdf0e10cSrcweir     {
693cdf0e10cSrcweir         if( rFmt.EqualsIgnoreCaseAscii( pInfo->mpVbaFormat ) )
694cdf0e10cSrcweir             break;
695cdf0e10cSrcweir         i++;
696cdf0e10cSrcweir     }
697cdf0e10cSrcweir     return pInfo;
698cdf0e10cSrcweir }
699cdf0e10cSrcweir 
700cdf0e10cSrcweir #define VBAFORMAT_GENERALDATE       "General Date"
701cdf0e10cSrcweir #define VBAFORMAT_C                 "c"
702cdf0e10cSrcweir #define VBAFORMAT_N                 "n"
703cdf0e10cSrcweir #define VBAFORMAT_NN                "nn"
704cdf0e10cSrcweir #define VBAFORMAT_W                 "w"
705cdf0e10cSrcweir #define VBAFORMAT_Y                 "y"
706cdf0e10cSrcweir #define VBAFORMAT_LOWERCASE         "<"
707cdf0e10cSrcweir #define VBAFORMAT_UPPERCASE         ">"
708cdf0e10cSrcweir 
709cdf0e10cSrcweir // From methods1.cxx
710cdf0e10cSrcweir sal_Int16 implGetWeekDay( double aDate, bool bFirstDayParam = false, sal_Int16 nFirstDay = 0 );
711cdf0e10cSrcweir // from methods.cxx
712cdf0e10cSrcweir sal_Int16 implGetMinute( double dDate );
713cdf0e10cSrcweir sal_Int16 implGetDateYear( double aDate );
714cdf0e10cSrcweir sal_Bool implDateSerial( sal_Int16 nYear, sal_Int16 nMonth, sal_Int16 nDay, double& rdRet );
715cdf0e10cSrcweir 
Format(XubString & rRes,const XubString * pFmt) const716cdf0e10cSrcweir void SbxValue::Format( XubString& rRes, const XubString* pFmt ) const
717cdf0e10cSrcweir {
718cdf0e10cSrcweir     short nComma = 0;
719cdf0e10cSrcweir     double d = 0;
720cdf0e10cSrcweir 
721cdf0e10cSrcweir     // pflin, It is better to use SvNumberFormatter to handle the date/time/number format.
722cdf0e10cSrcweir     // the SvNumberFormatter output is mostly compatible with
723*29d7882dSmseidel     // VBA output besides the AOO-BASIC output
724cdf0e10cSrcweir     if( pFmt && !SbxBasicFormater::isBasicFormat( *pFmt ) )
725cdf0e10cSrcweir     {
726cdf0e10cSrcweir         String aStr = GetString();
727cdf0e10cSrcweir 
728b4cc7af6SAriel Constenla-Haile         SvtSysLocale aSysLocale;
729b4cc7af6SAriel Constenla-Haile         const CharClass& rCharClass = aSysLocale.GetCharClass();
730b4cc7af6SAriel Constenla-Haile 
731cdf0e10cSrcweir         if( pFmt->EqualsIgnoreCaseAscii( VBAFORMAT_LOWERCASE ) )
732cdf0e10cSrcweir         {
733b4cc7af6SAriel Constenla-Haile             rCharClass.toLower( aStr );
734b4cc7af6SAriel Constenla-Haile             rRes = aStr;
735cdf0e10cSrcweir             return;
736cdf0e10cSrcweir         }
737cdf0e10cSrcweir         if( pFmt->EqualsIgnoreCaseAscii( VBAFORMAT_UPPERCASE ) )
738cdf0e10cSrcweir         {
739b4cc7af6SAriel Constenla-Haile             rCharClass.toUpper( aStr );
740b4cc7af6SAriel Constenla-Haile             rRes = aStr;
741cdf0e10cSrcweir             return;
742cdf0e10cSrcweir         }
743cdf0e10cSrcweir 
744cdf0e10cSrcweir         LanguageType eLangType = GetpApp()->GetSettings().GetLanguage();
745cdf0e10cSrcweir         com::sun::star::uno::Reference< com::sun::star::lang::XMultiServiceFactory >
746cdf0e10cSrcweir             xFactory = comphelper::getProcessServiceFactory();
747cdf0e10cSrcweir         SvNumberFormatter aFormatter( xFactory, eLangType );
748cdf0e10cSrcweir 
749cdf0e10cSrcweir         sal_uInt32 nIndex;
750cdf0e10cSrcweir         xub_StrLen nCheckPos = 0;
751cdf0e10cSrcweir         short nType;
752cdf0e10cSrcweir         double nNumber;
753cdf0e10cSrcweir         Color* pCol;
754cdf0e10cSrcweir 
755cdf0e10cSrcweir         sal_Bool bSuccess = aFormatter.IsNumberFormat( aStr, nIndex, nNumber );
756cdf0e10cSrcweir 
757cdf0e10cSrcweir         // number format, use SvNumberFormatter to handle it.
758cdf0e10cSrcweir         if( bSuccess )
759cdf0e10cSrcweir         {
760cdf0e10cSrcweir             String aFmtStr = *pFmt;
761cdf0e10cSrcweir             VbaFormatInfo* pInfo = getFormatInfo( aFmtStr );
762cdf0e10cSrcweir             if( pInfo && pInfo->meType != VBA_FORMAT_TYPE_NULL )
763cdf0e10cSrcweir             {
764cdf0e10cSrcweir                 if( pInfo->meType == VBA_FORMAT_TYPE_OFFSET )
765cdf0e10cSrcweir                 {
766cdf0e10cSrcweir                     nIndex = aFormatter.GetFormatIndex( pInfo->meOffset, eLangType );
767cdf0e10cSrcweir                 }
768cdf0e10cSrcweir                 else
769cdf0e10cSrcweir                 {
770cdf0e10cSrcweir                     aFmtStr.AssignAscii( pInfo->mpOOoFormat );
771cdf0e10cSrcweir                     aFormatter.PutandConvertEntry( aFmtStr, nCheckPos, nType, nIndex, LANGUAGE_ENGLISH, eLangType );
772cdf0e10cSrcweir                 }
773cdf0e10cSrcweir                 aFormatter.GetOutputString( nNumber, nIndex, rRes, &pCol );
774cdf0e10cSrcweir             }
775cdf0e10cSrcweir             else if( aFmtStr.EqualsIgnoreCaseAscii( VBAFORMAT_GENERALDATE )
776cdf0e10cSrcweir                     || aFmtStr.EqualsIgnoreCaseAscii( VBAFORMAT_C ))
777cdf0e10cSrcweir             {
778cdf0e10cSrcweir                 if( nNumber <=-1.0 || nNumber >= 1.0 )
779cdf0e10cSrcweir                 {
780cdf0e10cSrcweir                     // short date
781cdf0e10cSrcweir                     nIndex = aFormatter.GetFormatIndex( NF_DATE_SYSTEM_SHORT, eLangType );
782cdf0e10cSrcweir                     aFormatter.GetOutputString( nNumber, nIndex, rRes, &pCol );
783cdf0e10cSrcweir 
784cdf0e10cSrcweir                     // long time
785cdf0e10cSrcweir                     if( floor( nNumber ) != nNumber )
786cdf0e10cSrcweir                     {
787cdf0e10cSrcweir                         aFmtStr.AssignAscii( "H:MM:SS AM/PM" );
788cdf0e10cSrcweir                         aFormatter.PutandConvertEntry( aFmtStr, nCheckPos, nType, nIndex, LANGUAGE_ENGLISH, eLangType );
789cdf0e10cSrcweir                         String aTime;
790cdf0e10cSrcweir                         aFormatter.GetOutputString( nNumber, nIndex, aTime, &pCol );
791cdf0e10cSrcweir                         rRes.AppendAscii(" ");
792cdf0e10cSrcweir                         rRes += aTime;
793cdf0e10cSrcweir                     }
794cdf0e10cSrcweir                 }
795cdf0e10cSrcweir                 else
796cdf0e10cSrcweir                 {
797cdf0e10cSrcweir                     // long time only
798cdf0e10cSrcweir                     aFmtStr.AssignAscii( "H:MM:SS AM/PM" );
799cdf0e10cSrcweir                     aFormatter.PutandConvertEntry( aFmtStr, nCheckPos, nType, nIndex, LANGUAGE_ENGLISH, eLangType );
800cdf0e10cSrcweir                     aFormatter.GetOutputString( nNumber, nIndex, rRes, &pCol );
801cdf0e10cSrcweir                 }
802cdf0e10cSrcweir             }
803cdf0e10cSrcweir             else if( aFmtStr.EqualsIgnoreCaseAscii( VBAFORMAT_N )
804cdf0e10cSrcweir                     || aFmtStr.EqualsIgnoreCaseAscii( VBAFORMAT_NN ))
805cdf0e10cSrcweir             {
806cdf0e10cSrcweir                 sal_Int32 nMin = implGetMinute( nNumber );
807cdf0e10cSrcweir                 if( nMin < 10 && aFmtStr.EqualsIgnoreCaseAscii( VBAFORMAT_NN ) )
808cdf0e10cSrcweir                 {
809cdf0e10cSrcweir                     // Minute in two digits
810cdf0e10cSrcweir                      sal_Unicode* p = rRes.AllocBuffer( 2 );
811cdf0e10cSrcweir                      *p++ = '0';
812cdf0e10cSrcweir                      *p = sal_Unicode( '0' + nMin );
813cdf0e10cSrcweir                 }
814cdf0e10cSrcweir                 else
815cdf0e10cSrcweir                 {
816cdf0e10cSrcweir                     rRes = String::CreateFromInt32( nMin );
817cdf0e10cSrcweir                 }
818cdf0e10cSrcweir             }
819cdf0e10cSrcweir             else if( aFmtStr.EqualsIgnoreCaseAscii( VBAFORMAT_W ))
820cdf0e10cSrcweir             {
821cdf0e10cSrcweir                 sal_Int32 nWeekDay = implGetWeekDay( nNumber );
822cdf0e10cSrcweir                 rRes = String::CreateFromInt32( nWeekDay );
823cdf0e10cSrcweir             }
824cdf0e10cSrcweir             else if( aFmtStr.EqualsIgnoreCaseAscii( VBAFORMAT_Y ))
825cdf0e10cSrcweir             {
826cdf0e10cSrcweir                 sal_Int16 nYear = implGetDateYear( nNumber );
827cdf0e10cSrcweir                 double dBaseDate;
828cdf0e10cSrcweir                 implDateSerial( nYear, 1, 1, dBaseDate );
829cdf0e10cSrcweir                 sal_Int32 nYear32 = 1 + sal_Int32( nNumber - dBaseDate );
830cdf0e10cSrcweir                 rRes = String::CreateFromInt32( nYear32 );
831cdf0e10cSrcweir             }
832cdf0e10cSrcweir             else
833cdf0e10cSrcweir             {
834cdf0e10cSrcweir                 aFormatter.PutandConvertEntry( aFmtStr, nCheckPos, nType, nIndex, LANGUAGE_ENGLISH, eLangType );
835cdf0e10cSrcweir                 aFormatter.GetOutputString( nNumber, nIndex, rRes, &pCol );
836cdf0e10cSrcweir             }
837cdf0e10cSrcweir 
838cdf0e10cSrcweir             return;
839cdf0e10cSrcweir         }
840cdf0e10cSrcweir     }
841cdf0e10cSrcweir 
842cdf0e10cSrcweir     SbxDataType eType = GetType();
843cdf0e10cSrcweir     switch( eType )
844cdf0e10cSrcweir     {
845cdf0e10cSrcweir         case SbxCHAR:
846cdf0e10cSrcweir         case SbxBYTE:
847cdf0e10cSrcweir         case SbxINTEGER:
848cdf0e10cSrcweir         case SbxUSHORT:
849cdf0e10cSrcweir         case SbxLONG:
850cdf0e10cSrcweir         case SbxULONG:
851cdf0e10cSrcweir         case SbxINT:
852cdf0e10cSrcweir         case SbxUINT:
853cdf0e10cSrcweir         case SbxNULL:       // #45929 NULL mit durchschummeln
854cdf0e10cSrcweir             nComma = 0;     goto cvt;
855cdf0e10cSrcweir         case SbxSINGLE:
856cdf0e10cSrcweir             nComma = 6;     goto cvt;
857cdf0e10cSrcweir         case SbxDOUBLE:
858cdf0e10cSrcweir             nComma = 14;
859cdf0e10cSrcweir 
860cdf0e10cSrcweir         cvt:
861cdf0e10cSrcweir             if( eType != SbxNULL )
862cdf0e10cSrcweir                 d = GetDouble();
863cdf0e10cSrcweir 
864*29d7882dSmseidel             // #45355 weiterer Einsprungpunkt für isnumeric-String
865cdf0e10cSrcweir         cvt2:
866cdf0e10cSrcweir             if( pFmt )
867cdf0e10cSrcweir             {
868cdf0e10cSrcweir                 // hole die 'statischen' Daten f"ur Sbx
869cdf0e10cSrcweir                 SbxAppData* pData = GetSbxData_Impl();
870cdf0e10cSrcweir 
871cdf0e10cSrcweir                 LanguageType eLangType = GetpApp()->GetSettings().GetLanguage();
872cdf0e10cSrcweir                 if( pData->pBasicFormater )
873cdf0e10cSrcweir                 {
874cdf0e10cSrcweir                     if( pData->eBasicFormaterLangType != eLangType )
875cdf0e10cSrcweir                     {
876cdf0e10cSrcweir                         delete pData->pBasicFormater;
877cdf0e10cSrcweir                         pData->pBasicFormater = NULL;
878cdf0e10cSrcweir                     }
879cdf0e10cSrcweir                 }
880cdf0e10cSrcweir                 pData->eBasicFormaterLangType = eLangType;
881cdf0e10cSrcweir 
882cdf0e10cSrcweir                 // falls bisher noch kein BasicFormater-Objekt
883cdf0e10cSrcweir                 // existiert, so erzeuge dieses
884cdf0e10cSrcweir                 if( !pData->pBasicFormater )
885cdf0e10cSrcweir                 {
886cdf0e10cSrcweir                     SvtSysLocale aSysLocale;
887cdf0e10cSrcweir                     const LocaleDataWrapper& rData = aSysLocale.GetLocaleData();
888cdf0e10cSrcweir                     sal_Unicode cComma = rData.getNumDecimalSep().GetBuffer()[0];
889cdf0e10cSrcweir                     sal_Unicode c1000  = rData.getNumThousandSep().GetBuffer()[0];
890cdf0e10cSrcweir                     String aCurrencyStrg = rData.getCurrSymbol();
891cdf0e10cSrcweir 
892cdf0e10cSrcweir                     // Initialisierung des Basic-Formater-Hilfsobjekts:
893*29d7882dSmseidel                     // hole die Ressourcen für die vordefinierten Ausgaben
894*29d7882dSmseidel                     // des Format()-Befehls, z.B. für "On/Off".
895cdf0e10cSrcweir                     String aOnStrg = String( SbxValueFormatResId(
896cdf0e10cSrcweir                         STR_BASICKEY_FORMAT_ON ) );
897cdf0e10cSrcweir                     String aOffStrg = String( SbxValueFormatResId(
898cdf0e10cSrcweir                         STR_BASICKEY_FORMAT_OFF) );
899cdf0e10cSrcweir                     String aYesStrg = String( SbxValueFormatResId(
900cdf0e10cSrcweir                         STR_BASICKEY_FORMAT_YES) );
901cdf0e10cSrcweir                     String aNoStrg = String( SbxValueFormatResId(
902cdf0e10cSrcweir                         STR_BASICKEY_FORMAT_NO) );
903cdf0e10cSrcweir                     String aTrueStrg = String( SbxValueFormatResId(
904cdf0e10cSrcweir                         STR_BASICKEY_FORMAT_TRUE) );
905cdf0e10cSrcweir                     String aFalseStrg = String( SbxValueFormatResId(
906cdf0e10cSrcweir                         STR_BASICKEY_FORMAT_FALSE) );
907cdf0e10cSrcweir                     String aCurrencyFormatStrg = String( SbxValueFormatResId(
908cdf0e10cSrcweir                         STR_BASICKEY_FORMAT_CURRENCY) );
909cdf0e10cSrcweir                     // erzeuge das Basic-Formater-Objekt
910cdf0e10cSrcweir                     pData->pBasicFormater
911cdf0e10cSrcweir                         = new SbxBasicFormater( cComma,c1000,aOnStrg,aOffStrg,
912cdf0e10cSrcweir                                     aYesStrg,aNoStrg,aTrueStrg,aFalseStrg,
913cdf0e10cSrcweir                                     aCurrencyStrg,aCurrencyFormatStrg );
914cdf0e10cSrcweir                 }
915*29d7882dSmseidel                 // Bem.: Aus Performance-Gründen wird nur EIN BasicFormater-
916cdf0e10cSrcweir                 //    Objekt erzeugt und 'gespeichert', dadurch erspart man
917*29d7882dSmseidel                 //    sich das teure Ressourcen-Laden (für landesspezifische
918*29d7882dSmseidel                 //    vordefinierte Ausgaben, z.B. "On/Off") und die ständige
919cdf0e10cSrcweir                 //    String-Erzeugungs Operationen.
920cdf0e10cSrcweir                 // ABER: dadurch ist dieser Code NICHT multithreading f"ahig !
921cdf0e10cSrcweir 
922cdf0e10cSrcweir                 // hier gibt es Probleme mit ;;;Null, da diese Methode nur aufgerufen
923cdf0e10cSrcweir                 // wird, wenn der SbxValue eine Zahl ist !!!
924cdf0e10cSrcweir                 // dazu koennte: pData->pBasicFormater->BasicFormatNull( *pFmt ); aufgerufen werden !
925cdf0e10cSrcweir                 if( eType != SbxNULL )
926cdf0e10cSrcweir                 {
927cdf0e10cSrcweir                     rRes = pData->pBasicFormater->BasicFormat( d ,*pFmt );
928cdf0e10cSrcweir                 }
929cdf0e10cSrcweir                 else
930cdf0e10cSrcweir                 {
931cdf0e10cSrcweir                     rRes = pData->pBasicFormater->BasicFormatNull( *pFmt );
932cdf0e10cSrcweir                 }
933cdf0e10cSrcweir 
934cdf0e10cSrcweir                 // Die alte Implementierung:
935cdf0e10cSrcweir                 //old: printfmtnum( GetDouble(), rRes, *pFmt );
936cdf0e10cSrcweir             }
937cdf0e10cSrcweir             else
938cdf0e10cSrcweir             {
939cdf0e10cSrcweir                 ::rtl::OUString aTmpString( rRes );
940cdf0e10cSrcweir                 ImpCvtNum( GetDouble(), nComma, aTmpString );
941cdf0e10cSrcweir                 rRes = aTmpString;
942cdf0e10cSrcweir             }
943cdf0e10cSrcweir             break;
944cdf0e10cSrcweir         case SbxSTRING:
945cdf0e10cSrcweir             if( pFmt )
946cdf0e10cSrcweir             {
947cdf0e10cSrcweir                 // #45355 wenn es numerisch ist, muss gewandelt werden
948cdf0e10cSrcweir                 if( IsNumericRTL() )
949cdf0e10cSrcweir                 {
950cdf0e10cSrcweir                     ScanNumIntnl( GetString(), d, /*bSingle*/sal_False );
951cdf0e10cSrcweir                     goto cvt2;
952cdf0e10cSrcweir                 }
953cdf0e10cSrcweir                 else
954cdf0e10cSrcweir                 {
955cdf0e10cSrcweir                     // Sonst String-Formatierung
956cdf0e10cSrcweir                     printfmtstr( GetString(), rRes, *pFmt );
957cdf0e10cSrcweir                 }
958cdf0e10cSrcweir             }
959cdf0e10cSrcweir             else
960cdf0e10cSrcweir                 rRes = GetString();
961cdf0e10cSrcweir             break;
962cdf0e10cSrcweir         default:
963cdf0e10cSrcweir             rRes = GetString();
964cdf0e10cSrcweir     }
965cdf0e10cSrcweir }
966cdf0e10cSrcweir 
967*29d7882dSmseidel /* vim: set noet sw=4 ts=4: */
968