xref: /trunk/main/basic/source/runtime/step0.cxx (revision 9f813b30a249a2ed1e0f5e3ff3c0b88a8924b427)
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 
22e1f63238SAndrew Rist 
23cdf0e10cSrcweir 
24cdf0e10cSrcweir // MARKER(update_precomp.py): autogen include statement, do not remove
25cdf0e10cSrcweir #include "precompiled_basic.hxx"
26cdf0e10cSrcweir #include <vcl/msgbox.hxx>
27cdf0e10cSrcweir #include <tools/fsys.hxx>
28cdf0e10cSrcweir 
29cdf0e10cSrcweir #include "errobject.hxx"
30cdf0e10cSrcweir #include "runtime.hxx"
31cdf0e10cSrcweir #include "sbintern.hxx"
32cdf0e10cSrcweir #include "iosys.hxx"
33cdf0e10cSrcweir #include <sb.hrc>
34cdf0e10cSrcweir #include <basrid.hxx>
35cdf0e10cSrcweir #include "sbunoobj.hxx"
36cdf0e10cSrcweir #include "image.hxx"
37cdf0e10cSrcweir #include <com/sun/star/uno/Any.hxx>
38cdf0e10cSrcweir #include <com/sun/star/util/SearchOptions.hdl>
39cdf0e10cSrcweir #include <vcl/svapp.hxx>
40cdf0e10cSrcweir #include <unotools/textsearch.hxx>
41cdf0e10cSrcweir 
42cdf0e10cSrcweir Reference< XInterface > createComListener( const Any& aControlAny, const ::rtl::OUString& aVBAType,
43cdf0e10cSrcweir                                            const ::rtl::OUString& aPrefix, SbxObjectRef xScopeObj );
44cdf0e10cSrcweir 
45cdf0e10cSrcweir #include <algorithm>
46cdf0e10cSrcweir #include <hash_map>
47cdf0e10cSrcweir 
48cdf0e10cSrcweir SbxVariable* getDefaultProp( SbxVariable* pRef );
49cdf0e10cSrcweir 
50cdf0e10cSrcweir void SbiRuntime::StepNOP()
51cdf0e10cSrcweir {}
52cdf0e10cSrcweir 
53cdf0e10cSrcweir void SbiRuntime::StepArith( SbxOperator eOp )
54cdf0e10cSrcweir {
55cdf0e10cSrcweir     SbxVariableRef p1 = PopVar();
56cdf0e10cSrcweir     TOSMakeTemp();
57cdf0e10cSrcweir     SbxVariable* p2 = GetTOS();
58cdf0e10cSrcweir 
59cdf0e10cSrcweir 
60cdf0e10cSrcweir     // This could & should be moved to the MakeTempTOS() method in runtime.cxx
61cdf0e10cSrcweir     // In the code which this is cut'npaste from there is a check for a ref
62cdf0e10cSrcweir     // count != 1 based on which the copy of the SbxVariable is done.
63cdf0e10cSrcweir     // see orig code in MakeTempTOS ( and I'm not sure what the significance,
64cdf0e10cSrcweir     // of that is )
65cdf0e10cSrcweir     // here we alway seem to have a refcount of 1. Also it seems that
66cdf0e10cSrcweir     // MakeTempTOS is called for other operation, so I hold off for now
67cdf0e10cSrcweir     // until I have a better idea
68cdf0e10cSrcweir     if ( bVBAEnabled
69cdf0e10cSrcweir         && ( p2->GetType() == SbxOBJECT || p2->GetType() == SbxVARIANT )
70cdf0e10cSrcweir     )
71cdf0e10cSrcweir     {
72cdf0e10cSrcweir         SbxVariable* pDflt = getDefaultProp( p2 );
73cdf0e10cSrcweir         if ( pDflt )
74cdf0e10cSrcweir         {
75cdf0e10cSrcweir             pDflt->Broadcast( SBX_HINT_DATAWANTED );
76cdf0e10cSrcweir             // replacing new p2 on stack causes object pointed by
77cdf0e10cSrcweir             // pDft->pParent to be deleted, when p2->Compute() is
78cdf0e10cSrcweir             // called below pParent is accessed ( but its deleted )
79cdf0e10cSrcweir             // so set it to NULL now
80cdf0e10cSrcweir             pDflt->SetParent( NULL );
81cdf0e10cSrcweir             p2 = new SbxVariable( *pDflt );
82cdf0e10cSrcweir             p2->SetFlag( SBX_READWRITE );
83cdf0e10cSrcweir             refExprStk->Put( p2, nExprLvl - 1 );
84cdf0e10cSrcweir         }
85cdf0e10cSrcweir     }
86cdf0e10cSrcweir 
87cdf0e10cSrcweir     p2->ResetFlag( SBX_FIXED );
88cdf0e10cSrcweir     p2->Compute( eOp, *p1 );
89cdf0e10cSrcweir 
90cdf0e10cSrcweir     checkArithmeticOverflow( p2 );
91cdf0e10cSrcweir }
92cdf0e10cSrcweir 
93cdf0e10cSrcweir void SbiRuntime::StepUnary( SbxOperator eOp )
94cdf0e10cSrcweir {
95cdf0e10cSrcweir     TOSMakeTemp();
96cdf0e10cSrcweir     SbxVariable* p = GetTOS();
97cdf0e10cSrcweir     p->Compute( eOp, *p );
98cdf0e10cSrcweir }
99cdf0e10cSrcweir 
100cdf0e10cSrcweir void SbiRuntime::StepCompare( SbxOperator eOp )
101cdf0e10cSrcweir {
102cdf0e10cSrcweir     SbxVariableRef p1 = PopVar();
103cdf0e10cSrcweir     SbxVariableRef p2 = PopVar();
104cdf0e10cSrcweir 
105cdf0e10cSrcweir     // Make sure objects with default params have
106cdf0e10cSrcweir     // values ( and type ) set as appropriate
107cdf0e10cSrcweir     SbxDataType p1Type = p1->GetType();
108cdf0e10cSrcweir     SbxDataType p2Type = p2->GetType();
109cdf0e10cSrcweir     if ( p1Type == p2Type )
110cdf0e10cSrcweir     {
111cdf0e10cSrcweir         if ( p1Type == SbxEMPTY )
112cdf0e10cSrcweir         {
113cdf0e10cSrcweir             p1->Broadcast( SBX_HINT_DATAWANTED );
114cdf0e10cSrcweir             p2->Broadcast( SBX_HINT_DATAWANTED );
115cdf0e10cSrcweir         }
116cdf0e10cSrcweir         // if both sides are an object and have default props
117cdf0e10cSrcweir         // then we need to use the default props
118cdf0e10cSrcweir         // we don't need to worry if only one side ( lhs, rhs ) is an
119cdf0e10cSrcweir         // object ( object side will get coerced to correct type in
120cdf0e10cSrcweir         // Compare )
121cdf0e10cSrcweir         else if ( p1Type ==  SbxOBJECT )
122cdf0e10cSrcweir         {
123cdf0e10cSrcweir             SbxVariable* pDflt = getDefaultProp( p1 );
124cdf0e10cSrcweir             if ( pDflt )
125cdf0e10cSrcweir             {
126cdf0e10cSrcweir                 p1 = pDflt;
127cdf0e10cSrcweir                 p1->Broadcast( SBX_HINT_DATAWANTED );
128cdf0e10cSrcweir             }
129cdf0e10cSrcweir             pDflt = getDefaultProp( p2 );
130cdf0e10cSrcweir             if ( pDflt )
131cdf0e10cSrcweir             {
132cdf0e10cSrcweir                 p2 = pDflt;
133cdf0e10cSrcweir                 p2->Broadcast( SBX_HINT_DATAWANTED );
134cdf0e10cSrcweir             }
135cdf0e10cSrcweir         }
136cdf0e10cSrcweir 
137cdf0e10cSrcweir     }
138cdf0e10cSrcweir     static SbxVariable* pTRUE = NULL;
139cdf0e10cSrcweir     static SbxVariable* pFALSE = NULL;
140cdf0e10cSrcweir 
141cdf0e10cSrcweir     if( p2->Compare( eOp, *p1 ) )
142cdf0e10cSrcweir     {
143cdf0e10cSrcweir         if( !pTRUE )
144cdf0e10cSrcweir         {
145cdf0e10cSrcweir             pTRUE = new SbxVariable;
146cdf0e10cSrcweir             pTRUE->PutBool( sal_True );
147cdf0e10cSrcweir             pTRUE->AddRef();
148cdf0e10cSrcweir         }
149cdf0e10cSrcweir         PushVar( pTRUE );
150cdf0e10cSrcweir     }
151cdf0e10cSrcweir     else
152cdf0e10cSrcweir     {
153cdf0e10cSrcweir         if( !pFALSE )
154cdf0e10cSrcweir         {
155cdf0e10cSrcweir             pFALSE = new SbxVariable;
156cdf0e10cSrcweir             pFALSE->PutBool( sal_False );
157cdf0e10cSrcweir             pFALSE->AddRef();
158cdf0e10cSrcweir         }
159cdf0e10cSrcweir         PushVar( pFALSE );
160cdf0e10cSrcweir     }
161cdf0e10cSrcweir }
162cdf0e10cSrcweir 
163cdf0e10cSrcweir void SbiRuntime::StepEXP()      { StepArith( SbxEXP );      }
164cdf0e10cSrcweir void SbiRuntime::StepMUL()      { StepArith( SbxMUL );      }
165cdf0e10cSrcweir void SbiRuntime::StepDIV()      { StepArith( SbxDIV );      }
166cdf0e10cSrcweir void SbiRuntime::StepIDIV()     { StepArith( SbxIDIV );     }
167cdf0e10cSrcweir void SbiRuntime::StepMOD()      { StepArith( SbxMOD );      }
168cdf0e10cSrcweir void SbiRuntime::StepPLUS()     { StepArith( SbxPLUS );     }
169cdf0e10cSrcweir void SbiRuntime::StepMINUS()        { StepArith( SbxMINUS );    }
170cdf0e10cSrcweir void SbiRuntime::StepCAT()      { StepArith( SbxCAT );      }
171cdf0e10cSrcweir void SbiRuntime::StepAND()      { StepArith( SbxAND );      }
172cdf0e10cSrcweir void SbiRuntime::StepOR()       { StepArith( SbxOR );       }
173cdf0e10cSrcweir void SbiRuntime::StepXOR()      { StepArith( SbxXOR );      }
174cdf0e10cSrcweir void SbiRuntime::StepEQV()      { StepArith( SbxEQV );      }
175cdf0e10cSrcweir void SbiRuntime::StepIMP()      { StepArith( SbxIMP );      }
176cdf0e10cSrcweir 
177cdf0e10cSrcweir void SbiRuntime::StepNEG()      { StepUnary( SbxNEG );      }
178cdf0e10cSrcweir void SbiRuntime::StepNOT()      { StepUnary( SbxNOT );      }
179cdf0e10cSrcweir 
180cdf0e10cSrcweir void SbiRuntime::StepEQ()       { StepCompare( SbxEQ );     }
181cdf0e10cSrcweir void SbiRuntime::StepNE()       { StepCompare( SbxNE );     }
182cdf0e10cSrcweir void SbiRuntime::StepLT()       { StepCompare( SbxLT );     }
183cdf0e10cSrcweir void SbiRuntime::StepGT()       { StepCompare( SbxGT );     }
184cdf0e10cSrcweir void SbiRuntime::StepLE()       { StepCompare( SbxLE );     }
185cdf0e10cSrcweir void SbiRuntime::StepGE()       { StepCompare( SbxGE );     }
186cdf0e10cSrcweir 
187cdf0e10cSrcweir namespace
188cdf0e10cSrcweir {
189cdf0e10cSrcweir     bool NeedEsc(sal_Unicode cCode)
190cdf0e10cSrcweir     {
191cdf0e10cSrcweir         String sEsc(RTL_CONSTASCII_USTRINGPARAM(".^$+\\|{}()"));
192cdf0e10cSrcweir         return (STRING_NOTFOUND != sEsc.Search(cCode));
193cdf0e10cSrcweir     }
194cdf0e10cSrcweir 
195cdf0e10cSrcweir     String VBALikeToRegexp(const String &rIn)
196cdf0e10cSrcweir     {
197cdf0e10cSrcweir         String sResult;
198cdf0e10cSrcweir         const sal_Unicode *start = rIn.GetBuffer();
199cdf0e10cSrcweir         const sal_Unicode *end = start + rIn.Len();
200cdf0e10cSrcweir 
201cdf0e10cSrcweir         int seenright = 0;
202cdf0e10cSrcweir 
203cdf0e10cSrcweir         sResult.Append('^');
204cdf0e10cSrcweir 
205cdf0e10cSrcweir         while (start < end)
206cdf0e10cSrcweir         {
207cdf0e10cSrcweir             switch (*start)
208cdf0e10cSrcweir             {
209cdf0e10cSrcweir                 case '?':
210cdf0e10cSrcweir                     sResult.Append('.');
211cdf0e10cSrcweir                     start++;
212cdf0e10cSrcweir                     break;
213cdf0e10cSrcweir                 case '*':
214cdf0e10cSrcweir                     sResult.Append(String(RTL_CONSTASCII_USTRINGPARAM(".*")));
215cdf0e10cSrcweir                     start++;
216cdf0e10cSrcweir                     break;
217cdf0e10cSrcweir                 case '#':
218cdf0e10cSrcweir                     sResult.Append(String(RTL_CONSTASCII_USTRINGPARAM("[0-9]")));
219cdf0e10cSrcweir                     start++;
220cdf0e10cSrcweir                     break;
221cdf0e10cSrcweir                 case ']':
222cdf0e10cSrcweir                     sResult.Append('\\');
223cdf0e10cSrcweir                     sResult.Append(*start++);
224cdf0e10cSrcweir                     break;
225cdf0e10cSrcweir                 case '[':
226cdf0e10cSrcweir                     sResult.Append(*start++);
227cdf0e10cSrcweir                     seenright = 0;
228cdf0e10cSrcweir                     while (start < end && !seenright)
229cdf0e10cSrcweir                     {
230cdf0e10cSrcweir                         switch (*start)
231cdf0e10cSrcweir                         {
232cdf0e10cSrcweir                             case '[':
233cdf0e10cSrcweir                             case '?':
234cdf0e10cSrcweir                             case '*':
235cdf0e10cSrcweir                             sResult.Append('\\');
236cdf0e10cSrcweir                             sResult.Append(*start);
237cdf0e10cSrcweir                                 break;
238cdf0e10cSrcweir                             case ']':
239cdf0e10cSrcweir                             sResult.Append(*start);
240cdf0e10cSrcweir                                 seenright = 1;
241cdf0e10cSrcweir                                 break;
242cdf0e10cSrcweir                             case '!':
243cdf0e10cSrcweir                                 sResult.Append('^');
244cdf0e10cSrcweir                                 break;
245cdf0e10cSrcweir                             default:
246cdf0e10cSrcweir                             if (NeedEsc(*start))
247cdf0e10cSrcweir                                     sResult.Append('\\');
248cdf0e10cSrcweir                             sResult.Append(*start);
249cdf0e10cSrcweir                                 break;
250cdf0e10cSrcweir                         }
251cdf0e10cSrcweir                         start++;
252cdf0e10cSrcweir                     }
253cdf0e10cSrcweir                     break;
254cdf0e10cSrcweir                 default:
255cdf0e10cSrcweir                     if (NeedEsc(*start))
256cdf0e10cSrcweir                         sResult.Append('\\');
257cdf0e10cSrcweir                     sResult.Append(*start++);
258cdf0e10cSrcweir             }
259cdf0e10cSrcweir         }
260cdf0e10cSrcweir 
261cdf0e10cSrcweir         sResult.Append('$');
262cdf0e10cSrcweir 
263cdf0e10cSrcweir         return sResult;
264cdf0e10cSrcweir     }
265cdf0e10cSrcweir }
266cdf0e10cSrcweir 
267cdf0e10cSrcweir void SbiRuntime::StepLIKE()
268cdf0e10cSrcweir {
269cdf0e10cSrcweir     SbxVariableRef refVar1 = PopVar();
270cdf0e10cSrcweir     SbxVariableRef refVar2 = PopVar();
271cdf0e10cSrcweir 
272cdf0e10cSrcweir     String pattern = VBALikeToRegexp(refVar1->GetString());
273cdf0e10cSrcweir     String value = refVar2->GetString();
274cdf0e10cSrcweir 
275cdf0e10cSrcweir     com::sun::star::util::SearchOptions aSearchOpt;
276cdf0e10cSrcweir 
277cdf0e10cSrcweir     aSearchOpt.algorithmType = com::sun::star::util::SearchAlgorithms_REGEXP;
278cdf0e10cSrcweir 
279cdf0e10cSrcweir     aSearchOpt.Locale = Application::GetSettings().GetLocale();
280cdf0e10cSrcweir     aSearchOpt.searchString = pattern;
281cdf0e10cSrcweir 
282cdf0e10cSrcweir     int bTextMode(1);
283cdf0e10cSrcweir     bool bCompatibility = ( pINST && pINST->IsCompatibility() );
284cdf0e10cSrcweir     if( bCompatibility )
285cdf0e10cSrcweir         bTextMode = GetImageFlag( SBIMG_COMPARETEXT );
286cdf0e10cSrcweir 
287cdf0e10cSrcweir     if( bTextMode )
288cdf0e10cSrcweir         aSearchOpt.transliterateFlags |= com::sun::star::i18n::TransliterationModules_IGNORE_CASE;
289cdf0e10cSrcweir 
290cdf0e10cSrcweir     SbxVariable* pRes = new SbxVariable;
291cdf0e10cSrcweir     utl::TextSearch aSearch(aSearchOpt);
292cdf0e10cSrcweir     xub_StrLen nStart=0, nEnd=value.Len();
293cdf0e10cSrcweir     int bRes = aSearch.SearchFrwrd(value, &nStart, &nEnd);
294cdf0e10cSrcweir     pRes->PutBool( bRes != 0 );
295cdf0e10cSrcweir 
296cdf0e10cSrcweir     PushVar( pRes );
297cdf0e10cSrcweir }
298cdf0e10cSrcweir 
299cdf0e10cSrcweir // TOS und TOS-1 sind beides Objektvariable und enthalten den selben Pointer
300cdf0e10cSrcweir 
301cdf0e10cSrcweir void SbiRuntime::StepIS()
302cdf0e10cSrcweir {
303cdf0e10cSrcweir     SbxVariableRef refVar1 = PopVar();
304cdf0e10cSrcweir     SbxVariableRef refVar2 = PopVar();
305cdf0e10cSrcweir 
306cdf0e10cSrcweir     SbxDataType eType1 = refVar1->GetType();
307cdf0e10cSrcweir     SbxDataType eType2 = refVar2->GetType();
308cdf0e10cSrcweir     if ( eType1 == SbxEMPTY )
309cdf0e10cSrcweir     {
310cdf0e10cSrcweir         refVar1->Broadcast( SBX_HINT_DATAWANTED );
311cdf0e10cSrcweir         eType1 = refVar1->GetType();
312cdf0e10cSrcweir     }
313cdf0e10cSrcweir     if ( eType2 == SbxEMPTY )
314cdf0e10cSrcweir     {
315cdf0e10cSrcweir         refVar2->Broadcast( SBX_HINT_DATAWANTED );
316cdf0e10cSrcweir         eType2 = refVar2->GetType();
317cdf0e10cSrcweir     }
318cdf0e10cSrcweir 
319cdf0e10cSrcweir     sal_Bool bRes = sal_Bool( eType1 == SbxOBJECT && eType2 == SbxOBJECT );
320cdf0e10cSrcweir     if ( bVBAEnabled  && !bRes )
321cdf0e10cSrcweir         Error( SbERR_INVALID_USAGE_OBJECT );
322cdf0e10cSrcweir     bRes = ( bRes && refVar1->GetObject() == refVar2->GetObject() );
323cdf0e10cSrcweir     SbxVariable* pRes = new SbxVariable;
324cdf0e10cSrcweir     pRes->PutBool( bRes );
325cdf0e10cSrcweir     PushVar( pRes );
326cdf0e10cSrcweir }
327cdf0e10cSrcweir 
328cdf0e10cSrcweir // Aktualisieren des Wertes von TOS
329cdf0e10cSrcweir 
330cdf0e10cSrcweir void SbiRuntime::StepGET()
331cdf0e10cSrcweir {
332cdf0e10cSrcweir     SbxVariable* p = GetTOS();
333cdf0e10cSrcweir     p->Broadcast( SBX_HINT_DATAWANTED );
334cdf0e10cSrcweir }
335cdf0e10cSrcweir 
336cdf0e10cSrcweir // #67607 Uno-Structs kopieren
337cdf0e10cSrcweir inline void checkUnoStructCopy( SbxVariableRef& refVal, SbxVariableRef& refVar )
338cdf0e10cSrcweir {
339cdf0e10cSrcweir     SbxDataType eVarType = refVar->GetType();
340cdf0e10cSrcweir     if( eVarType != SbxOBJECT )
341cdf0e10cSrcweir         return;
342cdf0e10cSrcweir 
343cdf0e10cSrcweir     SbxObjectRef xValObj = (SbxObject*)refVal->GetObject();
344cdf0e10cSrcweir     if( !xValObj.Is() || xValObj->ISA(SbUnoAnyObject) )
345cdf0e10cSrcweir         return;
346cdf0e10cSrcweir 
347cdf0e10cSrcweir     // #115826: Exclude ProcedureProperties to avoid call to Property Get procedure
348cdf0e10cSrcweir     if( refVar->ISA(SbProcedureProperty) )
349cdf0e10cSrcweir         return;
350cdf0e10cSrcweir 
351cdf0e10cSrcweir     SbxObjectRef xVarObj = (SbxObject*)refVar->GetObject();
352cdf0e10cSrcweir     SbxDataType eValType = refVal->GetType();
353cdf0e10cSrcweir     if( eValType == SbxOBJECT && xVarObj == xValObj )
354cdf0e10cSrcweir     {
355cdf0e10cSrcweir         SbUnoObject* pUnoObj = PTR_CAST(SbUnoObject,(SbxObject*)xVarObj);
356cdf0e10cSrcweir         if( pUnoObj )
357cdf0e10cSrcweir         {
358cdf0e10cSrcweir             Any aAny = pUnoObj->getUnoAny();
359cdf0e10cSrcweir             if( aAny.getValueType().getTypeClass() == TypeClass_STRUCT )
360cdf0e10cSrcweir             {
361cdf0e10cSrcweir                 SbUnoObject* pNewUnoObj = new SbUnoObject( pUnoObj->GetName(), aAny );
362cdf0e10cSrcweir                 // #70324: ClassName uebernehmen
363cdf0e10cSrcweir                 pNewUnoObj->SetClassName( pUnoObj->GetClassName() );
364cdf0e10cSrcweir                 refVar->PutObject( pNewUnoObj );
365cdf0e10cSrcweir             }
366cdf0e10cSrcweir         }
367cdf0e10cSrcweir     }
368cdf0e10cSrcweir }
369cdf0e10cSrcweir 
370cdf0e10cSrcweir 
371cdf0e10cSrcweir // Ablage von TOS in TOS-1
372cdf0e10cSrcweir 
373cdf0e10cSrcweir void SbiRuntime::StepPUT()
374cdf0e10cSrcweir {
375cdf0e10cSrcweir     SbxVariableRef refVal = PopVar();
376cdf0e10cSrcweir     SbxVariableRef refVar = PopVar();
377cdf0e10cSrcweir     // Store auf die eigene Methode (innerhalb einer Function)?
378cdf0e10cSrcweir     sal_Bool bFlagsChanged = sal_False;
379cdf0e10cSrcweir     sal_uInt16 n = 0;
380cdf0e10cSrcweir     if( (SbxVariable*) refVar == (SbxVariable*) pMeth )
381cdf0e10cSrcweir     {
382cdf0e10cSrcweir         bFlagsChanged = sal_True;
383cdf0e10cSrcweir         n = refVar->GetFlags();
384cdf0e10cSrcweir         refVar->SetFlag( SBX_WRITE );
385cdf0e10cSrcweir     }
386cdf0e10cSrcweir 
387cdf0e10cSrcweir     // if left side arg is an object or variant and right handside isn't
388cdf0e10cSrcweir     // either an object or a variant then try and see if a default
389cdf0e10cSrcweir     // property exists.
390cdf0e10cSrcweir     // to use e.g. Range{"A1") = 34
391cdf0e10cSrcweir     // could equate to Range("A1").Value = 34
392cdf0e10cSrcweir     if ( bVBAEnabled )
393cdf0e10cSrcweir     {
394cdf0e10cSrcweir         if ( refVar->GetType() == SbxOBJECT  )
395cdf0e10cSrcweir         {
396cdf0e10cSrcweir             SbxVariable* pDflt = getDefaultProp( refVar );
397cdf0e10cSrcweir             if ( pDflt )
398cdf0e10cSrcweir                 refVar = pDflt;
399cdf0e10cSrcweir         }
400cdf0e10cSrcweir         if (  refVal->GetType() == SbxOBJECT  )
401cdf0e10cSrcweir         {
402cdf0e10cSrcweir             SbxVariable* pDflt = getDefaultProp( refVal );
403cdf0e10cSrcweir             if ( pDflt )
404cdf0e10cSrcweir                 refVal = pDflt;
405cdf0e10cSrcweir         }
406cdf0e10cSrcweir     }
407cdf0e10cSrcweir 
408cdf0e10cSrcweir     *refVar = *refVal;
409cdf0e10cSrcweir     // lhs is a property who's value is currently null
410cdf0e10cSrcweir     if ( !bVBAEnabled || ( bVBAEnabled && refVar->GetType() != SbxEMPTY ) )
411cdf0e10cSrcweir     // #67607 Uno-Structs kopieren
412cdf0e10cSrcweir         checkUnoStructCopy( refVal, refVar );
413cdf0e10cSrcweir     if( bFlagsChanged )
414cdf0e10cSrcweir         refVar->SetFlags( n );
415cdf0e10cSrcweir }
416cdf0e10cSrcweir 
417cdf0e10cSrcweir 
418cdf0e10cSrcweir // VBA Dim As New behavior handling, save init object information
419cdf0e10cSrcweir struct DimAsNewRecoverItem
420cdf0e10cSrcweir {
421cdf0e10cSrcweir     String          m_aObjClass;
422cdf0e10cSrcweir     String          m_aObjName;
423cdf0e10cSrcweir     SbxObject*      m_pObjParent;
424cdf0e10cSrcweir     SbModule*       m_pClassModule;
425cdf0e10cSrcweir 
426cdf0e10cSrcweir     DimAsNewRecoverItem( void )
427cdf0e10cSrcweir         : m_pObjParent( NULL )
428cdf0e10cSrcweir         , m_pClassModule( NULL )
429cdf0e10cSrcweir     {}
430cdf0e10cSrcweir 
431cdf0e10cSrcweir     DimAsNewRecoverItem( const String& rObjClass, const String& rObjName,
432cdf0e10cSrcweir         SbxObject* pObjParent, SbModule* pClassModule )
433cdf0e10cSrcweir             : m_aObjClass( rObjClass )
434cdf0e10cSrcweir             , m_aObjName( rObjName )
435cdf0e10cSrcweir             , m_pObjParent( pObjParent )
436cdf0e10cSrcweir             , m_pClassModule( pClassModule )
437cdf0e10cSrcweir     {}
438cdf0e10cSrcweir 
439cdf0e10cSrcweir };
440cdf0e10cSrcweir 
441cdf0e10cSrcweir 
442cdf0e10cSrcweir struct SbxVariablePtrHash
443cdf0e10cSrcweir {
444cdf0e10cSrcweir     size_t operator()( SbxVariable* pVar ) const
445cdf0e10cSrcweir         { return (size_t)pVar; }
446cdf0e10cSrcweir };
447cdf0e10cSrcweir 
448cdf0e10cSrcweir typedef std::hash_map< SbxVariable*, DimAsNewRecoverItem, SbxVariablePtrHash >  DimAsNewRecoverHash;
449cdf0e10cSrcweir 
450cdf0e10cSrcweir static DimAsNewRecoverHash      GaDimAsNewRecoverHash;
451cdf0e10cSrcweir 
452cdf0e10cSrcweir void removeDimAsNewRecoverItem( SbxVariable* pVar )
453cdf0e10cSrcweir {
454cdf0e10cSrcweir     DimAsNewRecoverHash::iterator it = GaDimAsNewRecoverHash.find( pVar );
455cdf0e10cSrcweir     if( it != GaDimAsNewRecoverHash.end() )
456cdf0e10cSrcweir         GaDimAsNewRecoverHash.erase( it );
457cdf0e10cSrcweir }
458cdf0e10cSrcweir 
459cdf0e10cSrcweir 
460cdf0e10cSrcweir // Speichern Objektvariable
461cdf0e10cSrcweir // Nicht-Objekt-Variable fuehren zu Fehlern
462cdf0e10cSrcweir 
463cdf0e10cSrcweir static const char pCollectionStr[] = "Collection";
464cdf0e10cSrcweir 
465cdf0e10cSrcweir void SbiRuntime::StepSET_Impl( SbxVariableRef& refVal, SbxVariableRef& refVar, bool bHandleDefaultProp )
466cdf0e10cSrcweir {
467cdf0e10cSrcweir     // #67733 Typen mit Array-Flag sind auch ok
468cdf0e10cSrcweir 
469cdf0e10cSrcweir     // Check var, !object is no error for sure if, only if type is fixed
470cdf0e10cSrcweir     SbxDataType eVarType = refVar->GetType();
471cdf0e10cSrcweir     if( !bHandleDefaultProp && eVarType != SbxOBJECT && !(eVarType & SbxARRAY) && refVar->IsFixed() )
472cdf0e10cSrcweir     {
473cdf0e10cSrcweir         Error( SbERR_INVALID_USAGE_OBJECT );
474cdf0e10cSrcweir         return;
475cdf0e10cSrcweir     }
476cdf0e10cSrcweir 
477cdf0e10cSrcweir     // Check value, !object is no error for sure if, only if type is fixed
478cdf0e10cSrcweir     SbxDataType eValType = refVal->GetType();
479cdf0e10cSrcweir //  bool bGetValObject = false;
480cdf0e10cSrcweir     if( !bHandleDefaultProp && eValType != SbxOBJECT && !(eValType & SbxARRAY) && refVal->IsFixed() )
481cdf0e10cSrcweir     {
482cdf0e10cSrcweir         Error( SbERR_INVALID_USAGE_OBJECT );
483cdf0e10cSrcweir         return;
484cdf0e10cSrcweir     }
485cdf0e10cSrcweir 
486cdf0e10cSrcweir     // Getting in here causes problems with objects with default properties
487cdf0e10cSrcweir     // if they are SbxEMPTY I guess
488cdf0e10cSrcweir     if ( !bHandleDefaultProp || ( bHandleDefaultProp && eValType == SbxOBJECT ) )
489cdf0e10cSrcweir     {
490cdf0e10cSrcweir     // Auf refVal GetObject fuer Collections ausloesen
491cdf0e10cSrcweir         SbxBase* pObjVarObj = refVal->GetObject();
492cdf0e10cSrcweir         if( pObjVarObj )
493cdf0e10cSrcweir         {
494cdf0e10cSrcweir             SbxVariableRef refObjVal = PTR_CAST(SbxObject,pObjVarObj);
495cdf0e10cSrcweir 
496cdf0e10cSrcweir             // #67733 Typen mit Array-Flag sind auch ok
497cdf0e10cSrcweir             if( refObjVal )
498cdf0e10cSrcweir                 refVal = refObjVal;
499cdf0e10cSrcweir             else if( !(eValType & SbxARRAY) )
500cdf0e10cSrcweir                 refVal = NULL;
501cdf0e10cSrcweir         }
502cdf0e10cSrcweir     }
503cdf0e10cSrcweir 
504cdf0e10cSrcweir     // #52896 Wenn Uno-Sequences bzw. allgemein Arrays einer als
505cdf0e10cSrcweir     // Object deklarierten Variable zugewiesen werden, kann hier
506cdf0e10cSrcweir     // refVal ungueltig sein!
507cdf0e10cSrcweir     if( !refVal )
508cdf0e10cSrcweir     {
509cdf0e10cSrcweir         Error( SbERR_INVALID_USAGE_OBJECT );
510cdf0e10cSrcweir     }
511cdf0e10cSrcweir     else
512cdf0e10cSrcweir     {
513cdf0e10cSrcweir         // Store auf die eigene Methode (innerhalb einer Function)?
514cdf0e10cSrcweir         sal_Bool bFlagsChanged = sal_False;
515cdf0e10cSrcweir         sal_uInt16 n = 0;
516cdf0e10cSrcweir         if( (SbxVariable*) refVar == (SbxVariable*) pMeth )
517cdf0e10cSrcweir         {
518cdf0e10cSrcweir             bFlagsChanged = sal_True;
519cdf0e10cSrcweir             n = refVar->GetFlags();
520cdf0e10cSrcweir             refVar->SetFlag( SBX_WRITE );
521cdf0e10cSrcweir         }
522cdf0e10cSrcweir         SbProcedureProperty* pProcProperty = PTR_CAST(SbProcedureProperty,(SbxVariable*)refVar);
523cdf0e10cSrcweir         if( pProcProperty )
524cdf0e10cSrcweir             pProcProperty->setSet( true );
525cdf0e10cSrcweir 
526cdf0e10cSrcweir         if ( bHandleDefaultProp )
527cdf0e10cSrcweir         {
528cdf0e10cSrcweir             // get default properties for lhs & rhs where necessary
529cdf0e10cSrcweir             // SbxVariable* defaultProp = NULL; unused variable
530cdf0e10cSrcweir             bool bLHSHasDefaultProp = false;
531cdf0e10cSrcweir             // LHS try determine if a default prop exists
532cdf0e10cSrcweir             if ( refVar->GetType() == SbxOBJECT )
533cdf0e10cSrcweir             {
534cdf0e10cSrcweir                 SbxVariable* pDflt = getDefaultProp( refVar );
535cdf0e10cSrcweir                 if ( pDflt )
536cdf0e10cSrcweir                 {
537cdf0e10cSrcweir                     refVar = pDflt;
538cdf0e10cSrcweir                     bLHSHasDefaultProp = true;
539cdf0e10cSrcweir                 }
540cdf0e10cSrcweir             }
541cdf0e10cSrcweir             // RHS only get a default prop is the rhs has one
542cdf0e10cSrcweir             if (  refVal->GetType() == SbxOBJECT )
543cdf0e10cSrcweir             {
544cdf0e10cSrcweir                 // check if lhs is a null object
545cdf0e10cSrcweir                 // if it is then use the object not the default property
546cdf0e10cSrcweir                 SbxObject* pObj = NULL;
547cdf0e10cSrcweir 
548cdf0e10cSrcweir 
549cdf0e10cSrcweir                 pObj = PTR_CAST(SbxObject,(SbxVariable*)refVar);
550cdf0e10cSrcweir 
551cdf0e10cSrcweir                 // calling GetObject on a SbxEMPTY variable raises
552cdf0e10cSrcweir                 // object not set errors, make sure its an Object
553cdf0e10cSrcweir                 if ( !pObj && refVar->GetType() == SbxOBJECT )
554cdf0e10cSrcweir                 {
555cdf0e10cSrcweir                     SbxBase* pObjVarObj = refVar->GetObject();
556cdf0e10cSrcweir                     pObj = PTR_CAST(SbxObject,pObjVarObj);
557cdf0e10cSrcweir                 }
558cdf0e10cSrcweir                 SbxVariable* pDflt = NULL;
559cdf0e10cSrcweir                 if ( pObj || bLHSHasDefaultProp )
560cdf0e10cSrcweir                     // lhs is either a valid object || or has a defaultProp
561cdf0e10cSrcweir                     pDflt = getDefaultProp( refVal );
562cdf0e10cSrcweir                 if ( pDflt )
563cdf0e10cSrcweir                     refVal = pDflt;
564cdf0e10cSrcweir             }
565cdf0e10cSrcweir         }
566cdf0e10cSrcweir 
567cdf0e10cSrcweir         // Handle Dim As New
568cdf0e10cSrcweir         sal_Bool bDimAsNew = bVBAEnabled && refVar->IsSet( SBX_DIM_AS_NEW );
569cdf0e10cSrcweir         SbxBaseRef xPrevVarObj;
570cdf0e10cSrcweir         if( bDimAsNew )
571cdf0e10cSrcweir             xPrevVarObj = refVar->GetObject();
572cdf0e10cSrcweir 
573cdf0e10cSrcweir         // Handle withevents
574cdf0e10cSrcweir         sal_Bool bWithEvents = refVar->IsSet( SBX_WITH_EVENTS );
575cdf0e10cSrcweir         if ( bWithEvents )
576cdf0e10cSrcweir         {
577cdf0e10cSrcweir             Reference< XInterface > xComListener;
578cdf0e10cSrcweir 
579cdf0e10cSrcweir             SbxBase* pObj = refVal->GetObject();
580cdf0e10cSrcweir             SbUnoObject* pUnoObj = (pObj != NULL) ? PTR_CAST(SbUnoObject,pObj) : NULL;
581cdf0e10cSrcweir             if( pUnoObj != NULL )
582cdf0e10cSrcweir             {
583cdf0e10cSrcweir                 Any aControlAny = pUnoObj->getUnoAny();
584cdf0e10cSrcweir                 String aDeclareClassName = refVar->GetDeclareClassName();
585cdf0e10cSrcweir                 ::rtl::OUString aVBAType = aDeclareClassName;
586cdf0e10cSrcweir                 ::rtl::OUString aPrefix = refVar->GetName();
587cdf0e10cSrcweir                 SbxObjectRef xScopeObj = refVar->GetParent();
588cdf0e10cSrcweir                 xComListener = createComListener( aControlAny, aVBAType, aPrefix, xScopeObj );
589cdf0e10cSrcweir 
590cdf0e10cSrcweir                 refVal->SetDeclareClassName( aDeclareClassName );
591cdf0e10cSrcweir                 refVal->SetComListener( xComListener, &rBasic );        // Hold reference
592cdf0e10cSrcweir             }
593cdf0e10cSrcweir 
594cdf0e10cSrcweir             *refVar = *refVal;
595cdf0e10cSrcweir         }
596cdf0e10cSrcweir         else
597cdf0e10cSrcweir         {
598cdf0e10cSrcweir             *refVar = *refVal;
599cdf0e10cSrcweir         }
600cdf0e10cSrcweir 
601cdf0e10cSrcweir         if ( bDimAsNew )
602cdf0e10cSrcweir         {
603cdf0e10cSrcweir             if( !refVar->ISA(SbxObject) )
604cdf0e10cSrcweir             {
605cdf0e10cSrcweir                 SbxBase* pValObjBase = refVal->GetObject();
606cdf0e10cSrcweir                 if( pValObjBase == NULL )
607cdf0e10cSrcweir                 {
608cdf0e10cSrcweir                     if( xPrevVarObj.Is() )
609cdf0e10cSrcweir                     {
610cdf0e10cSrcweir                         // Object is overwritten with NULL, instantiate init object
611cdf0e10cSrcweir                         DimAsNewRecoverHash::iterator it = GaDimAsNewRecoverHash.find( refVar );
612cdf0e10cSrcweir                         if( it != GaDimAsNewRecoverHash.end() )
613cdf0e10cSrcweir                         {
614cdf0e10cSrcweir                             const DimAsNewRecoverItem& rItem = it->second;
615cdf0e10cSrcweir                             if( rItem.m_pClassModule != NULL )
616cdf0e10cSrcweir                             {
617cdf0e10cSrcweir                                 SbClassModuleObject* pNewObj = new SbClassModuleObject( rItem.m_pClassModule );
618cdf0e10cSrcweir                                 pNewObj->SetName( rItem.m_aObjName );
619cdf0e10cSrcweir                                 pNewObj->SetParent( rItem.m_pObjParent );
620cdf0e10cSrcweir                                 refVar->PutObject( pNewObj );
621cdf0e10cSrcweir                             }
622cdf0e10cSrcweir                             else if( rItem.m_aObjClass.EqualsIgnoreCaseAscii( pCollectionStr ) )
623cdf0e10cSrcweir                             {
624cdf0e10cSrcweir                                 BasicCollection* pNewCollection = new BasicCollection( String( RTL_CONSTASCII_USTRINGPARAM(pCollectionStr) ) );
625cdf0e10cSrcweir                                 pNewCollection->SetName( rItem.m_aObjName );
626cdf0e10cSrcweir                                 pNewCollection->SetParent( rItem.m_pObjParent );
627cdf0e10cSrcweir                                 refVar->PutObject( pNewCollection );
628cdf0e10cSrcweir                             }
629cdf0e10cSrcweir                         }
630cdf0e10cSrcweir                     }
631cdf0e10cSrcweir                 }
632cdf0e10cSrcweir                 else
633cdf0e10cSrcweir                 {
634cdf0e10cSrcweir                     // Does old value exist?
635cdf0e10cSrcweir                     bool bFirstInit = !xPrevVarObj.Is();
636cdf0e10cSrcweir                     if( bFirstInit )
637cdf0e10cSrcweir                     {
638cdf0e10cSrcweir                         // Store information to instantiate object later
639cdf0e10cSrcweir                         SbxObject* pValObj = PTR_CAST(SbxObject,pValObjBase);
640cdf0e10cSrcweir                         if( pValObj != NULL )
641cdf0e10cSrcweir                         {
642cdf0e10cSrcweir                             String aObjClass = pValObj->GetClassName();
643cdf0e10cSrcweir 
644cdf0e10cSrcweir                             SbClassModuleObject* pClassModuleObj = PTR_CAST(SbClassModuleObject,pValObjBase);
645cdf0e10cSrcweir                             if( pClassModuleObj != NULL )
646cdf0e10cSrcweir                             {
647cdf0e10cSrcweir                                 SbModule* pClassModule = pClassModuleObj->getClassModule();
648cdf0e10cSrcweir                                 GaDimAsNewRecoverHash[refVar] =
649cdf0e10cSrcweir                                     DimAsNewRecoverItem( aObjClass, pValObj->GetName(), pValObj->GetParent(), pClassModule );
650cdf0e10cSrcweir                             }
651cdf0e10cSrcweir                             else if( aObjClass.EqualsIgnoreCaseAscii( "Collection" ) )
652cdf0e10cSrcweir                             {
653cdf0e10cSrcweir                                 GaDimAsNewRecoverHash[refVar] =
654cdf0e10cSrcweir                                     DimAsNewRecoverItem( aObjClass, pValObj->GetName(), pValObj->GetParent(), NULL );
655cdf0e10cSrcweir                             }
656cdf0e10cSrcweir                         }
657cdf0e10cSrcweir                     }
658cdf0e10cSrcweir                 }
659cdf0e10cSrcweir             }
660cdf0e10cSrcweir         }
661cdf0e10cSrcweir 
662cdf0e10cSrcweir 
663cdf0e10cSrcweir         // lhs is a property who's value is currently (Empty e.g. no broadcast yet)
664cdf0e10cSrcweir         // in this case if there is a default prop involved the value of the
665cdf0e10cSrcweir         // default property may infact be void so the type will also be SbxEMPTY
666cdf0e10cSrcweir         // in this case we do not want to call checkUnoStructCopy 'cause that will
667cdf0e10cSrcweir         // cause an error also
668cdf0e10cSrcweir         if ( !bHandleDefaultProp || ( bHandleDefaultProp && ( refVar->GetType() != SbxEMPTY ) ) )
669cdf0e10cSrcweir         // #67607 Uno-Structs kopieren
670cdf0e10cSrcweir             checkUnoStructCopy( refVal, refVar );
671cdf0e10cSrcweir         if( bFlagsChanged )
672cdf0e10cSrcweir             refVar->SetFlags( n );
673cdf0e10cSrcweir     }
674cdf0e10cSrcweir }
675cdf0e10cSrcweir 
676cdf0e10cSrcweir void SbiRuntime::StepSET()
677cdf0e10cSrcweir {
678cdf0e10cSrcweir     SbxVariableRef refVal = PopVar();
679cdf0e10cSrcweir     SbxVariableRef refVar = PopVar();
680cdf0e10cSrcweir     StepSET_Impl( refVal, refVar, bVBAEnabled ); // this is really assigment
681cdf0e10cSrcweir }
682cdf0e10cSrcweir 
683cdf0e10cSrcweir void SbiRuntime::StepVBASET()
684cdf0e10cSrcweir {
685cdf0e10cSrcweir     SbxVariableRef refVal = PopVar();
686cdf0e10cSrcweir     SbxVariableRef refVar = PopVar();
687cdf0e10cSrcweir     // don't handle default property
688cdf0e10cSrcweir     StepSET_Impl( refVal, refVar, false ); // set obj = something
689cdf0e10cSrcweir }
690cdf0e10cSrcweir 
691cdf0e10cSrcweir 
692cdf0e10cSrcweir // JSM 07.10.95
693cdf0e10cSrcweir void SbiRuntime::StepLSET()
694cdf0e10cSrcweir {
695cdf0e10cSrcweir     SbxVariableRef refVal = PopVar();
696cdf0e10cSrcweir     SbxVariableRef refVar = PopVar();
697cdf0e10cSrcweir     if( refVar->GetType() != SbxSTRING
698cdf0e10cSrcweir      || refVal->GetType() != SbxSTRING )
699cdf0e10cSrcweir         Error( SbERR_INVALID_USAGE_OBJECT );
700cdf0e10cSrcweir     else
701cdf0e10cSrcweir     {
702cdf0e10cSrcweir         // Store auf die eigene Methode (innerhalb einer Function)?
703cdf0e10cSrcweir         sal_uInt16 n = refVar->GetFlags();
704cdf0e10cSrcweir         if( (SbxVariable*) refVar == (SbxVariable*) pMeth )
705cdf0e10cSrcweir             refVar->SetFlag( SBX_WRITE );
706cdf0e10cSrcweir         String aRefVarString = refVar->GetString();
707cdf0e10cSrcweir         String aRefValString = refVal->GetString();
708cdf0e10cSrcweir 
709cdf0e10cSrcweir         sal_uInt16 nVarStrLen = aRefVarString.Len();
710cdf0e10cSrcweir         sal_uInt16 nValStrLen = aRefValString.Len();
711cdf0e10cSrcweir         String aNewStr;
712cdf0e10cSrcweir         if( nVarStrLen > nValStrLen )
713cdf0e10cSrcweir         {
714cdf0e10cSrcweir             aRefVarString.Fill(nVarStrLen,' ');
715cdf0e10cSrcweir             aNewStr  = aRefValString.Copy( 0, nValStrLen );
716cdf0e10cSrcweir             aNewStr += aRefVarString.Copy( nValStrLen, nVarStrLen - nValStrLen );
717cdf0e10cSrcweir         }
718cdf0e10cSrcweir         else
719cdf0e10cSrcweir         {
720cdf0e10cSrcweir             aNewStr = aRefValString.Copy( 0, nVarStrLen );
721cdf0e10cSrcweir         }
722cdf0e10cSrcweir 
723cdf0e10cSrcweir         refVar->PutString( aNewStr );
724cdf0e10cSrcweir         refVar->SetFlags( n );
725cdf0e10cSrcweir     }
726cdf0e10cSrcweir }
727cdf0e10cSrcweir 
728cdf0e10cSrcweir // JSM 07.10.95
729cdf0e10cSrcweir void SbiRuntime::StepRSET()
730cdf0e10cSrcweir {
731cdf0e10cSrcweir     SbxVariableRef refVal = PopVar();
732cdf0e10cSrcweir     SbxVariableRef refVar = PopVar();
733cdf0e10cSrcweir     if( refVar->GetType() != SbxSTRING
734cdf0e10cSrcweir      || refVal->GetType() != SbxSTRING )
735cdf0e10cSrcweir         Error( SbERR_INVALID_USAGE_OBJECT );
736cdf0e10cSrcweir     else
737cdf0e10cSrcweir     {
738cdf0e10cSrcweir         // Store auf die eigene Methode (innerhalb einer Function)?
739cdf0e10cSrcweir         sal_uInt16 n = refVar->GetFlags();
740cdf0e10cSrcweir         if( (SbxVariable*) refVar == (SbxVariable*) pMeth )
741cdf0e10cSrcweir             refVar->SetFlag( SBX_WRITE );
742cdf0e10cSrcweir         String aRefVarString = refVar->GetString();
743cdf0e10cSrcweir         String aRefValString = refVal->GetString();
744cdf0e10cSrcweir 
745cdf0e10cSrcweir         sal_uInt16 nPos = 0;
746cdf0e10cSrcweir         sal_uInt16 nVarStrLen = aRefVarString.Len();
747cdf0e10cSrcweir         if( nVarStrLen > aRefValString.Len() )
748cdf0e10cSrcweir         {
749cdf0e10cSrcweir             aRefVarString.Fill(nVarStrLen,' ');
750cdf0e10cSrcweir             nPos = nVarStrLen - aRefValString.Len();
751cdf0e10cSrcweir         }
752cdf0e10cSrcweir         aRefVarString  = aRefVarString.Copy( 0, nPos );
753cdf0e10cSrcweir         aRefVarString += aRefValString.Copy( 0, nVarStrLen - nPos );
754cdf0e10cSrcweir         refVar->PutString(aRefVarString);
755cdf0e10cSrcweir 
756cdf0e10cSrcweir         refVar->SetFlags( n );
757cdf0e10cSrcweir     }
758cdf0e10cSrcweir }
759cdf0e10cSrcweir 
760cdf0e10cSrcweir // Ablage von TOS in TOS-1, dann ReadOnly-Bit setzen
761cdf0e10cSrcweir 
762cdf0e10cSrcweir void SbiRuntime::StepPUTC()
763cdf0e10cSrcweir {
764cdf0e10cSrcweir     SbxVariableRef refVal = PopVar();
765cdf0e10cSrcweir     SbxVariableRef refVar = PopVar();
766cdf0e10cSrcweir     refVar->SetFlag( SBX_WRITE );
767cdf0e10cSrcweir     *refVar = *refVal;
768cdf0e10cSrcweir     refVar->ResetFlag( SBX_WRITE );
769cdf0e10cSrcweir     refVar->SetFlag( SBX_CONST );
770cdf0e10cSrcweir }
771cdf0e10cSrcweir 
772cdf0e10cSrcweir // DIM
773cdf0e10cSrcweir // TOS = Variable fuer das Array mit Dimensionsangaben als Parameter
774cdf0e10cSrcweir 
775cdf0e10cSrcweir void SbiRuntime::StepDIM()
776cdf0e10cSrcweir {
777cdf0e10cSrcweir     SbxVariableRef refVar = PopVar();
778cdf0e10cSrcweir     DimImpl( refVar );
779cdf0e10cSrcweir }
780cdf0e10cSrcweir 
781cdf0e10cSrcweir // #56204 DIM-Funktionalitaet in Hilfsmethode auslagern (step0.cxx)
782cdf0e10cSrcweir void SbiRuntime::DimImpl( SbxVariableRef refVar )
783cdf0e10cSrcweir {
784cdf0e10cSrcweir     SbxArray* pDims = refVar->GetParameters();
785cdf0e10cSrcweir     // Muss eine gerade Anzahl Argumente haben
786cdf0e10cSrcweir     // Man denke daran, dass Arg[0] nicht zaehlt!
787cdf0e10cSrcweir     if( pDims && !( pDims->Count() & 1 ) )
788cdf0e10cSrcweir         StarBASIC::FatalError( SbERR_INTERNAL_ERROR );
789cdf0e10cSrcweir     else
790cdf0e10cSrcweir     {
791cdf0e10cSrcweir         SbxDataType eType = refVar->IsFixed() ? refVar->GetType() : SbxVARIANT;
792cdf0e10cSrcweir         SbxDimArray* pArray = new SbxDimArray( eType );
793cdf0e10cSrcweir         // AB 2.4.1996, auch Arrays ohne Dimensionsangaben zulassen (VB-komp.)
794cdf0e10cSrcweir         if( pDims )
795cdf0e10cSrcweir         {
796cdf0e10cSrcweir             refVar->ResetFlag( SBX_VAR_TO_DIM );
797cdf0e10cSrcweir 
798cdf0e10cSrcweir             for( sal_uInt16 i = 1; i < pDims->Count(); )
799cdf0e10cSrcweir             {
800cdf0e10cSrcweir                 sal_Int32 lb = pDims->Get( i++ )->GetLong();
801cdf0e10cSrcweir                 sal_Int32 ub = pDims->Get( i++ )->GetLong();
802cdf0e10cSrcweir                 if( ub < lb )
803cdf0e10cSrcweir                     Error( SbERR_OUT_OF_RANGE ), ub = lb;
804cdf0e10cSrcweir                 pArray->AddDim32( lb, ub );
805cdf0e10cSrcweir                 if ( lb != ub )
806cdf0e10cSrcweir                     pArray->setHasFixedSize( true );
807cdf0e10cSrcweir             }
808cdf0e10cSrcweir         }
809cdf0e10cSrcweir         else
810cdf0e10cSrcweir         {
811cdf0e10cSrcweir             // #62867 Beim Anlegen eines Arrays der Laenge 0 wie bei
812cdf0e10cSrcweir             // Uno-Sequences der Laenge 0 eine Dimension anlegen
813cdf0e10cSrcweir             pArray->unoAddDim( 0, -1 );
814cdf0e10cSrcweir         }
815cdf0e10cSrcweir         sal_uInt16 nSavFlags = refVar->GetFlags();
816cdf0e10cSrcweir         refVar->ResetFlag( SBX_FIXED );
817cdf0e10cSrcweir         refVar->PutObject( pArray );
818cdf0e10cSrcweir         refVar->SetFlags( nSavFlags );
819cdf0e10cSrcweir         refVar->SetParameters( NULL );
820cdf0e10cSrcweir     }
821cdf0e10cSrcweir }
822cdf0e10cSrcweir 
823cdf0e10cSrcweir // REDIM
824cdf0e10cSrcweir // TOS  = Variable fuer das Array
825cdf0e10cSrcweir // argv = Dimensionsangaben
826cdf0e10cSrcweir 
827cdf0e10cSrcweir void SbiRuntime::StepREDIM()
828cdf0e10cSrcweir {
829cdf0e10cSrcweir     // Im Moment ist es nichts anderes als Dim, da doppeltes Dim
830cdf0e10cSrcweir     // bereits vom Compiler erkannt wird.
831cdf0e10cSrcweir     StepDIM();
832cdf0e10cSrcweir }
833cdf0e10cSrcweir 
834cdf0e10cSrcweir 
835cdf0e10cSrcweir // Helper function for StepREDIMP
836cdf0e10cSrcweir void implCopyDimArray( SbxDimArray* pNewArray, SbxDimArray* pOldArray, short nMaxDimIndex,
837cdf0e10cSrcweir     short nActualDim, sal_Int32* pActualIndices, sal_Int32* pLowerBounds, sal_Int32* pUpperBounds )
838cdf0e10cSrcweir {
839cdf0e10cSrcweir     sal_Int32& ri = pActualIndices[nActualDim];
840cdf0e10cSrcweir     for( ri = pLowerBounds[nActualDim] ; ri <= pUpperBounds[nActualDim] ; ri++ )
841cdf0e10cSrcweir     {
842cdf0e10cSrcweir         if( nActualDim < nMaxDimIndex )
843cdf0e10cSrcweir         {
844cdf0e10cSrcweir             implCopyDimArray( pNewArray, pOldArray, nMaxDimIndex, nActualDim + 1,
845cdf0e10cSrcweir                 pActualIndices, pLowerBounds, pUpperBounds );
846cdf0e10cSrcweir         }
847cdf0e10cSrcweir         else
848cdf0e10cSrcweir         {
849cdf0e10cSrcweir             SbxVariable* pSource = pOldArray->Get32( pActualIndices );
850cdf0e10cSrcweir             SbxVariable* pDest   = pNewArray->Get32( pActualIndices );
851cdf0e10cSrcweir             if( pSource && pDest )
852cdf0e10cSrcweir                 *pDest = *pSource;
853cdf0e10cSrcweir         }
854cdf0e10cSrcweir     }
855cdf0e10cSrcweir }
856cdf0e10cSrcweir 
857cdf0e10cSrcweir // REDIM PRESERVE
858cdf0e10cSrcweir // TOS  = Variable fuer das Array
859cdf0e10cSrcweir // argv = Dimensionsangaben
860cdf0e10cSrcweir 
861cdf0e10cSrcweir void SbiRuntime::StepREDIMP()
862cdf0e10cSrcweir {
863cdf0e10cSrcweir     SbxVariableRef refVar = PopVar();
864cdf0e10cSrcweir     DimImpl( refVar );
865cdf0e10cSrcweir 
866cdf0e10cSrcweir     // Now check, if we can copy from the old array
867cdf0e10cSrcweir     if( refRedimpArray.Is() )
868cdf0e10cSrcweir     {
869cdf0e10cSrcweir         SbxBase* pElemObj = refVar->GetObject();
870cdf0e10cSrcweir         SbxDimArray* pNewArray = PTR_CAST(SbxDimArray,pElemObj);
871cdf0e10cSrcweir         SbxDimArray* pOldArray = (SbxDimArray*)(SbxArray*)refRedimpArray;
872cdf0e10cSrcweir         if( pNewArray )
873cdf0e10cSrcweir         {
874cdf0e10cSrcweir             short nDimsNew = pNewArray->GetDims();
875cdf0e10cSrcweir             short nDimsOld = pOldArray->GetDims();
876cdf0e10cSrcweir             short nDims = nDimsNew;
877cdf0e10cSrcweir             sal_Bool bRangeError = sal_False;
878cdf0e10cSrcweir 
879cdf0e10cSrcweir             // Store dims to use them for copying later
880cdf0e10cSrcweir             sal_Int32* pLowerBounds = new sal_Int32[nDims];
881cdf0e10cSrcweir             sal_Int32* pUpperBounds = new sal_Int32[nDims];
882cdf0e10cSrcweir             sal_Int32* pActualIndices = new sal_Int32[nDims];
883cdf0e10cSrcweir 
884cdf0e10cSrcweir             if( nDimsOld != nDimsNew )
885cdf0e10cSrcweir             {
886cdf0e10cSrcweir                 bRangeError = sal_True;
887cdf0e10cSrcweir             }
888cdf0e10cSrcweir             else
889cdf0e10cSrcweir             {
890cdf0e10cSrcweir                 // Compare bounds
891cdf0e10cSrcweir                 for( short i = 1 ; i <= nDims ; i++ )
892cdf0e10cSrcweir                 {
893cdf0e10cSrcweir                     sal_Int32 lBoundNew, uBoundNew;
894cdf0e10cSrcweir                     sal_Int32 lBoundOld, uBoundOld;
895cdf0e10cSrcweir                     pNewArray->GetDim32( i, lBoundNew, uBoundNew );
896cdf0e10cSrcweir                     pOldArray->GetDim32( i, lBoundOld, uBoundOld );
897cdf0e10cSrcweir 
898cdf0e10cSrcweir                     /* #69094 Allow all dimensions to be changed
899cdf0e10cSrcweir                        although Visual Basic is not able to do so.
900cdf0e10cSrcweir                     // All bounds but the last have to be the same
901cdf0e10cSrcweir                     if( i < nDims && ( lBoundNew != lBoundOld || uBoundNew != uBoundOld ) )
902cdf0e10cSrcweir                     {
903cdf0e10cSrcweir                         bRangeError = sal_True;
904cdf0e10cSrcweir                         break;
905cdf0e10cSrcweir                     }
906cdf0e10cSrcweir                     else
907cdf0e10cSrcweir                     */
908cdf0e10cSrcweir                     {
909cdf0e10cSrcweir                         // #69094: if( i == nDims )
910cdf0e10cSrcweir                         {
911cdf0e10cSrcweir                             lBoundNew = std::max( lBoundNew, lBoundOld );
912cdf0e10cSrcweir                             uBoundNew = std::min( uBoundNew, uBoundOld );
913cdf0e10cSrcweir                         }
914cdf0e10cSrcweir                         short j = i - 1;
915cdf0e10cSrcweir                         pActualIndices[j] = pLowerBounds[j] = lBoundNew;
916cdf0e10cSrcweir                         pUpperBounds[j] = uBoundNew;
917cdf0e10cSrcweir                     }
918cdf0e10cSrcweir                 }
919cdf0e10cSrcweir             }
920cdf0e10cSrcweir 
921cdf0e10cSrcweir             if( bRangeError )
922cdf0e10cSrcweir             {
923cdf0e10cSrcweir                 StarBASIC::Error( SbERR_OUT_OF_RANGE );
924cdf0e10cSrcweir             }
925cdf0e10cSrcweir             else
926cdf0e10cSrcweir             {
927cdf0e10cSrcweir                 // Copy data from old array by going recursively through all dimensions
928cdf0e10cSrcweir                 // (It would be faster to work on the flat internal data array of an
929cdf0e10cSrcweir                 // SbyArray but this solution is clearer and easier)
930cdf0e10cSrcweir                 implCopyDimArray( pNewArray, pOldArray, nDims - 1,
931cdf0e10cSrcweir                     0, pActualIndices, pLowerBounds, pUpperBounds );
932cdf0e10cSrcweir             }
933cdf0e10cSrcweir 
934cdf0e10cSrcweir             delete[] pUpperBounds;
935cdf0e10cSrcweir             delete[] pLowerBounds;
936cdf0e10cSrcweir             delete[] pActualIndices;
937cdf0e10cSrcweir             refRedimpArray = NULL;
938cdf0e10cSrcweir         }
939cdf0e10cSrcweir     }
940cdf0e10cSrcweir 
941cdf0e10cSrcweir     //StarBASIC::FatalError( SbERR_NOT_IMPLEMENTED );
942cdf0e10cSrcweir }
943cdf0e10cSrcweir 
944cdf0e10cSrcweir // REDIM_COPY
945cdf0e10cSrcweir // TOS  = Array-Variable, Reference to array is copied
946cdf0e10cSrcweir //        Variable is cleared as in ERASE
947cdf0e10cSrcweir 
948cdf0e10cSrcweir void SbiRuntime::StepREDIMP_ERASE()
949cdf0e10cSrcweir {
950cdf0e10cSrcweir     SbxVariableRef refVar = PopVar();
951cdf0e10cSrcweir     SbxDataType eType = refVar->GetType();
952cdf0e10cSrcweir     if( eType & SbxARRAY )
953cdf0e10cSrcweir     {
954cdf0e10cSrcweir         SbxBase* pElemObj = refVar->GetObject();
955cdf0e10cSrcweir         SbxDimArray* pDimArray = PTR_CAST(SbxDimArray,pElemObj);
956cdf0e10cSrcweir         if( pDimArray )
957cdf0e10cSrcweir         {
958cdf0e10cSrcweir             refRedimpArray = pDimArray;
959cdf0e10cSrcweir         }
960cdf0e10cSrcweir 
961cdf0e10cSrcweir         // As in ERASE
962cdf0e10cSrcweir         sal_uInt16 nSavFlags = refVar->GetFlags();
963cdf0e10cSrcweir         refVar->ResetFlag( SBX_FIXED );
964cdf0e10cSrcweir         refVar->SetType( SbxDataType(eType & 0x0FFF) );
965cdf0e10cSrcweir         refVar->SetFlags( nSavFlags );
966cdf0e10cSrcweir         refVar->Clear();
967cdf0e10cSrcweir     }
968cdf0e10cSrcweir     else
969cdf0e10cSrcweir     if( refVar->IsFixed() )
970cdf0e10cSrcweir         refVar->Clear();
971cdf0e10cSrcweir     else
972cdf0e10cSrcweir         refVar->SetType( SbxEMPTY );
973cdf0e10cSrcweir }
974cdf0e10cSrcweir 
975cdf0e10cSrcweir void lcl_clearImpl( SbxVariableRef& refVar, SbxDataType& eType )
976cdf0e10cSrcweir {
977cdf0e10cSrcweir     sal_uInt16 nSavFlags = refVar->GetFlags();
978cdf0e10cSrcweir     refVar->ResetFlag( SBX_FIXED );
979cdf0e10cSrcweir     refVar->SetType( SbxDataType(eType & 0x0FFF) );
980cdf0e10cSrcweir     refVar->SetFlags( nSavFlags );
981cdf0e10cSrcweir     refVar->Clear();
982cdf0e10cSrcweir }
983cdf0e10cSrcweir 
984cdf0e10cSrcweir void lcl_eraseImpl( SbxVariableRef& refVar, bool bVBAEnabled )
985cdf0e10cSrcweir {
986cdf0e10cSrcweir     SbxDataType eType = refVar->GetType();
987cdf0e10cSrcweir     if( eType & SbxARRAY )
988cdf0e10cSrcweir     {
989cdf0e10cSrcweir         if ( bVBAEnabled )
990cdf0e10cSrcweir         {
991cdf0e10cSrcweir             SbxBase* pElemObj = refVar->GetObject();
992cdf0e10cSrcweir             SbxDimArray* pDimArray = PTR_CAST(SbxDimArray,pElemObj);
993cdf0e10cSrcweir             bool bClearValues = true;
994cdf0e10cSrcweir             if( pDimArray )
995cdf0e10cSrcweir             {
996cdf0e10cSrcweir                 if ( pDimArray->hasFixedSize() )
997cdf0e10cSrcweir                 {
998cdf0e10cSrcweir                     // Clear all Value(s)
999cdf0e10cSrcweir                     pDimArray->SbxArray::Clear();
1000cdf0e10cSrcweir                     bClearValues = false;
1001cdf0e10cSrcweir                 }
1002cdf0e10cSrcweir                 else
1003cdf0e10cSrcweir                     pDimArray->Clear(); // clear Dims
1004cdf0e10cSrcweir             }
1005cdf0e10cSrcweir             if ( bClearValues )
1006cdf0e10cSrcweir             {
1007cdf0e10cSrcweir                 SbxArray* pArray = PTR_CAST(SbxArray,pElemObj);
1008cdf0e10cSrcweir                 if ( pArray )
1009cdf0e10cSrcweir                     pArray->Clear();
1010cdf0e10cSrcweir             }
1011cdf0e10cSrcweir         }
1012cdf0e10cSrcweir         else
1013cdf0e10cSrcweir         // AB 2.4.1996
1014cdf0e10cSrcweir         // Arrays haben bei Erase nach VB ein recht komplexes Verhalten. Hier
1015cdf0e10cSrcweir         // werden zunaechst nur die Typ-Probleme bei REDIM (#26295) beseitigt:
1016cdf0e10cSrcweir         // Typ hart auf den Array-Typ setzen, da eine Variable mit Array
1017cdf0e10cSrcweir         // SbxOBJECT ist. Bei REDIM entsteht dann ein SbxOBJECT-Array und
1018cdf0e10cSrcweir         // der ursruengliche Typ geht verloren -> Laufzeitfehler
1019cdf0e10cSrcweir             lcl_clearImpl( refVar, eType );
1020cdf0e10cSrcweir     }
1021cdf0e10cSrcweir     else
1022cdf0e10cSrcweir     if( refVar->IsFixed() )
1023cdf0e10cSrcweir         refVar->Clear();
1024cdf0e10cSrcweir     else
1025cdf0e10cSrcweir         refVar->SetType( SbxEMPTY );
1026cdf0e10cSrcweir }
1027cdf0e10cSrcweir 
1028cdf0e10cSrcweir // Variable loeschen
1029cdf0e10cSrcweir // TOS = Variable
1030cdf0e10cSrcweir 
1031cdf0e10cSrcweir void SbiRuntime::StepERASE()
1032cdf0e10cSrcweir {
1033cdf0e10cSrcweir     SbxVariableRef refVar = PopVar();
1034cdf0e10cSrcweir     lcl_eraseImpl( refVar, bVBAEnabled );
1035cdf0e10cSrcweir }
1036cdf0e10cSrcweir 
1037cdf0e10cSrcweir void SbiRuntime::StepERASE_CLEAR()
1038cdf0e10cSrcweir {
1039cdf0e10cSrcweir     SbxVariableRef refVar = PopVar();
1040cdf0e10cSrcweir     lcl_eraseImpl( refVar, bVBAEnabled );
1041cdf0e10cSrcweir     SbxDataType eType = refVar->GetType();
1042cdf0e10cSrcweir     lcl_clearImpl( refVar, eType );
1043cdf0e10cSrcweir }
1044cdf0e10cSrcweir 
1045cdf0e10cSrcweir void SbiRuntime::StepARRAYACCESS()
1046cdf0e10cSrcweir {
1047cdf0e10cSrcweir     if( !refArgv )
1048cdf0e10cSrcweir         StarBASIC::FatalError( SbERR_INTERNAL_ERROR );
1049cdf0e10cSrcweir     SbxVariableRef refVar = PopVar();
1050cdf0e10cSrcweir     refVar->SetParameters( refArgv );
1051cdf0e10cSrcweir     PopArgv();
1052cdf0e10cSrcweir     PushVar( CheckArray( refVar ) );
1053cdf0e10cSrcweir }
1054cdf0e10cSrcweir 
1055cdf0e10cSrcweir void SbiRuntime::StepBYVAL()
1056cdf0e10cSrcweir {
1057cdf0e10cSrcweir     // Copy variable on stack to break call by reference
1058cdf0e10cSrcweir     SbxVariableRef pVar = PopVar();
1059cdf0e10cSrcweir     SbxDataType t = pVar->GetType();
1060cdf0e10cSrcweir 
1061cdf0e10cSrcweir     SbxVariable* pCopyVar = new SbxVariable( t );
1062cdf0e10cSrcweir     pCopyVar->SetFlag( SBX_READWRITE );
1063cdf0e10cSrcweir     *pCopyVar = *pVar;
1064cdf0e10cSrcweir 
1065cdf0e10cSrcweir     PushVar( pCopyVar );
1066cdf0e10cSrcweir }
1067cdf0e10cSrcweir 
1068cdf0e10cSrcweir // Einrichten eines Argvs
1069cdf0e10cSrcweir // nOp1 bleibt so -> 1. Element ist Returnwert
1070cdf0e10cSrcweir 
1071cdf0e10cSrcweir void SbiRuntime::StepARGC()
1072cdf0e10cSrcweir {
1073cdf0e10cSrcweir     PushArgv();
1074cdf0e10cSrcweir     refArgv = new SbxArray;
1075cdf0e10cSrcweir     nArgc = 1;
1076cdf0e10cSrcweir }
1077cdf0e10cSrcweir 
1078cdf0e10cSrcweir // Speichern eines Arguments in Argv
1079cdf0e10cSrcweir 
1080cdf0e10cSrcweir void SbiRuntime::StepARGV()
1081cdf0e10cSrcweir {
1082cdf0e10cSrcweir     if( !refArgv )
1083cdf0e10cSrcweir         StarBASIC::FatalError( SbERR_INTERNAL_ERROR );
1084cdf0e10cSrcweir     else
1085cdf0e10cSrcweir     {
1086cdf0e10cSrcweir         SbxVariableRef pVal = PopVar();
1087cdf0e10cSrcweir 
1088cdf0e10cSrcweir         // Before fix of #94916:
1089cdf0e10cSrcweir         // if( pVal->ISA(SbxMethod) || pVal->ISA(SbxProperty) )
1090cdf0e10cSrcweir         if( pVal->ISA(SbxMethod) || pVal->ISA(SbUnoProperty) || pVal->ISA(SbProcedureProperty) )
1091cdf0e10cSrcweir         {
1092cdf0e10cSrcweir             // Methoden und Properties evaluieren!
1093cdf0e10cSrcweir             SbxVariable* pRes = new SbxVariable( *pVal );
1094cdf0e10cSrcweir             pVal = pRes;
1095cdf0e10cSrcweir         }
1096cdf0e10cSrcweir         refArgv->Put( pVal, nArgc++ );
1097cdf0e10cSrcweir     }
1098cdf0e10cSrcweir }
1099cdf0e10cSrcweir 
1100cdf0e10cSrcweir // Input to Variable. Die Variable ist auf TOS und wird
1101cdf0e10cSrcweir // anschliessend entfernt.
1102cdf0e10cSrcweir 
1103cdf0e10cSrcweir void SbiRuntime::StepINPUT()
1104cdf0e10cSrcweir {
1105cdf0e10cSrcweir     String s;
1106cdf0e10cSrcweir     char ch = 0;
1107cdf0e10cSrcweir     SbError err;
1108cdf0e10cSrcweir     // Skip whitespace
1109cdf0e10cSrcweir     while( ( err = pIosys->GetError() ) == 0 )
1110cdf0e10cSrcweir     {
1111cdf0e10cSrcweir         ch = pIosys->Read();
1112cdf0e10cSrcweir         if( ch != ' ' && ch != '\t' && ch != '\n' )
1113cdf0e10cSrcweir             break;
1114cdf0e10cSrcweir     }
1115cdf0e10cSrcweir     if( !err )
1116cdf0e10cSrcweir     {
1117cdf0e10cSrcweir         // Scan until comma or whitespace
1118cdf0e10cSrcweir         char sep = ( ch == '"' ) ? ch : 0;
1119cdf0e10cSrcweir         if( sep ) ch = pIosys->Read();
1120cdf0e10cSrcweir         while( ( err = pIosys->GetError() ) == 0 )
1121cdf0e10cSrcweir         {
1122cdf0e10cSrcweir             if( ch == sep )
1123cdf0e10cSrcweir             {
1124cdf0e10cSrcweir                 ch = pIosys->Read();
1125cdf0e10cSrcweir                 if( ch != sep )
1126cdf0e10cSrcweir                     break;
1127cdf0e10cSrcweir             }
1128cdf0e10cSrcweir             else if( !sep && (ch == ',' || ch == '\n') )
1129cdf0e10cSrcweir                 break;
1130cdf0e10cSrcweir             s += ch;
1131cdf0e10cSrcweir             ch = pIosys->Read();
1132cdf0e10cSrcweir         }
1133cdf0e10cSrcweir         // skip whitespace
1134cdf0e10cSrcweir         if( ch == ' ' || ch == '\t' )
1135cdf0e10cSrcweir           while( ( err = pIosys->GetError() ) == 0 )
1136cdf0e10cSrcweir         {
1137cdf0e10cSrcweir             if( ch != ' ' && ch != '\t' && ch != '\n' )
1138cdf0e10cSrcweir                 break;
1139cdf0e10cSrcweir             ch = pIosys->Read();
1140cdf0e10cSrcweir         }
1141cdf0e10cSrcweir     }
1142cdf0e10cSrcweir     if( !err )
1143cdf0e10cSrcweir     {
1144cdf0e10cSrcweir         SbxVariableRef pVar = GetTOS();
1145cdf0e10cSrcweir         // Zuerst versuchen, die Variable mit einem numerischen Wert
1146cdf0e10cSrcweir         // zu fuellen, dann mit einem Stringwert
1147cdf0e10cSrcweir         if( !pVar->IsFixed() || pVar->IsNumeric() )
1148cdf0e10cSrcweir         {
1149cdf0e10cSrcweir             sal_uInt16 nLen = 0;
1150cdf0e10cSrcweir             if( !pVar->Scan( s, &nLen ) )
1151cdf0e10cSrcweir             {
1152cdf0e10cSrcweir                 err = SbxBase::GetError();
1153cdf0e10cSrcweir                 SbxBase::ResetError();
1154cdf0e10cSrcweir             }
1155cdf0e10cSrcweir             // Der Wert muss komplett eingescant werden
1156cdf0e10cSrcweir             else if( nLen != s.Len() && !pVar->PutString( s ) )
1157cdf0e10cSrcweir             {
1158cdf0e10cSrcweir                 err = SbxBase::GetError();
1159cdf0e10cSrcweir                 SbxBase::ResetError();
1160cdf0e10cSrcweir             }
1161cdf0e10cSrcweir             else if( nLen != s.Len() && pVar->IsNumeric() )
1162cdf0e10cSrcweir             {
1163cdf0e10cSrcweir                 err = SbxBase::GetError();
1164cdf0e10cSrcweir                 SbxBase::ResetError();
1165cdf0e10cSrcweir                 if( !err )
1166cdf0e10cSrcweir                     err = SbERR_CONVERSION;
1167cdf0e10cSrcweir             }
1168cdf0e10cSrcweir         }
1169cdf0e10cSrcweir         else
1170cdf0e10cSrcweir         {
1171cdf0e10cSrcweir             pVar->PutString( s );
1172cdf0e10cSrcweir             err = SbxBase::GetError();
1173cdf0e10cSrcweir             SbxBase::ResetError();
1174cdf0e10cSrcweir         }
1175cdf0e10cSrcweir     }
1176cdf0e10cSrcweir     if( err == SbERR_USER_ABORT )
1177cdf0e10cSrcweir         Error( err );
1178cdf0e10cSrcweir     else if( err )
1179cdf0e10cSrcweir     {
1180cdf0e10cSrcweir         if( pRestart && !pIosys->GetChannel() )
1181cdf0e10cSrcweir         {
1182cdf0e10cSrcweir             BasResId aId( IDS_SBERR_START + 4 );
1183cdf0e10cSrcweir             String aMsg( aId );
1184cdf0e10cSrcweir 
1185*9f813b30Smseidel             //****** DON'T CHECK IN, TEST ONLY *******
1186*9f813b30Smseidel             //****** DON'T CHECK IN, TEST ONLY *******
1187cdf0e10cSrcweir             // ErrorBox( NULL, WB_OK, aMsg ).Execute();
1188*9f813b30Smseidel             //****** DON'T CHECK IN, TEST ONLY *******
1189*9f813b30Smseidel             //****** DON'T CHECK IN, TEST ONLY *******
1190cdf0e10cSrcweir 
1191cdf0e10cSrcweir             pCode = pRestart;
1192cdf0e10cSrcweir         }
1193cdf0e10cSrcweir         else
1194cdf0e10cSrcweir             Error( err );
1195cdf0e10cSrcweir     }
1196cdf0e10cSrcweir     else
1197cdf0e10cSrcweir     {
1198cdf0e10cSrcweir         // pIosys->ResetChannel();
1199cdf0e10cSrcweir         PopVar();
1200cdf0e10cSrcweir     }
1201cdf0e10cSrcweir }
1202cdf0e10cSrcweir 
1203cdf0e10cSrcweir // Line Input to Variable. Die Variable ist auf TOS und wird
1204cdf0e10cSrcweir // anschliessend entfernt.
1205cdf0e10cSrcweir 
1206cdf0e10cSrcweir void SbiRuntime::StepLINPUT()
1207cdf0e10cSrcweir {
1208cdf0e10cSrcweir     ByteString aInput;
1209cdf0e10cSrcweir     pIosys->Read( aInput );
1210cdf0e10cSrcweir     Error( pIosys->GetError() );
1211cdf0e10cSrcweir     SbxVariableRef p = PopVar();
1212cdf0e10cSrcweir     p->PutString( String( aInput, gsl_getSystemTextEncoding() ) );
1213cdf0e10cSrcweir     // pIosys->ResetChannel();
1214cdf0e10cSrcweir }
1215cdf0e10cSrcweir 
1216cdf0e10cSrcweir // Programmende
1217cdf0e10cSrcweir 
1218cdf0e10cSrcweir void SbiRuntime::StepSTOP()
1219cdf0e10cSrcweir {
1220cdf0e10cSrcweir     pInst->Stop();
1221cdf0e10cSrcweir }
1222cdf0e10cSrcweir 
1223cdf0e10cSrcweir // FOR-Variable initialisieren
1224cdf0e10cSrcweir 
1225cdf0e10cSrcweir void SbiRuntime::StepINITFOR()
1226cdf0e10cSrcweir {
1227cdf0e10cSrcweir     PushFor();
1228cdf0e10cSrcweir }
1229cdf0e10cSrcweir 
1230cdf0e10cSrcweir void SbiRuntime::StepINITFOREACH()
1231cdf0e10cSrcweir {
1232cdf0e10cSrcweir     PushForEach();
1233cdf0e10cSrcweir }
1234cdf0e10cSrcweir 
1235cdf0e10cSrcweir // FOR-Variable inkrementieren
1236cdf0e10cSrcweir 
1237cdf0e10cSrcweir void SbiRuntime::StepNEXT()
1238cdf0e10cSrcweir {
1239cdf0e10cSrcweir     if( !pForStk )
1240cdf0e10cSrcweir     {
1241cdf0e10cSrcweir         StarBASIC::FatalError( SbERR_INTERNAL_ERROR );
1242cdf0e10cSrcweir         return;
1243cdf0e10cSrcweir     }
1244cdf0e10cSrcweir     if( pForStk->eForType == FOR_TO )
1245cdf0e10cSrcweir         pForStk->refVar->Compute( SbxPLUS, *pForStk->refInc );
1246cdf0e10cSrcweir }
1247cdf0e10cSrcweir 
1248cdf0e10cSrcweir // Anfang CASE: TOS in CASE-Stack
1249cdf0e10cSrcweir 
1250cdf0e10cSrcweir void SbiRuntime::StepCASE()
1251cdf0e10cSrcweir {
1252cdf0e10cSrcweir     if( !refCaseStk.Is() )
1253cdf0e10cSrcweir         refCaseStk = new SbxArray;
1254cdf0e10cSrcweir     SbxVariableRef xVar = PopVar();
1255cdf0e10cSrcweir     refCaseStk->Put( xVar, refCaseStk->Count() );
1256cdf0e10cSrcweir }
1257cdf0e10cSrcweir 
1258cdf0e10cSrcweir // Ende CASE: Variable freigeben
1259cdf0e10cSrcweir 
1260cdf0e10cSrcweir void SbiRuntime::StepENDCASE()
1261cdf0e10cSrcweir {
1262cdf0e10cSrcweir     if( !refCaseStk || !refCaseStk->Count() )
1263cdf0e10cSrcweir         StarBASIC::FatalError( SbERR_INTERNAL_ERROR );
1264cdf0e10cSrcweir     else
1265cdf0e10cSrcweir         refCaseStk->Remove( refCaseStk->Count() - 1 );
1266cdf0e10cSrcweir }
1267cdf0e10cSrcweir 
1268cdf0e10cSrcweir // Standard-Fehlerbehandlung
1269cdf0e10cSrcweir 
1270cdf0e10cSrcweir void SbiRuntime::StepSTDERROR()
1271cdf0e10cSrcweir {
1272cdf0e10cSrcweir     pError = NULL; bError = sal_True;
1273cdf0e10cSrcweir     pInst->aErrorMsg = String();
1274cdf0e10cSrcweir     pInst->nErr = 0L;
1275cdf0e10cSrcweir     pInst->nErl = 0;
1276cdf0e10cSrcweir     nError = 0L;
1277cdf0e10cSrcweir     SbxErrObject::getUnoErrObject()->Clear();
1278cdf0e10cSrcweir }
1279cdf0e10cSrcweir 
1280cdf0e10cSrcweir void SbiRuntime::StepNOERROR()
1281cdf0e10cSrcweir {
1282cdf0e10cSrcweir     pInst->aErrorMsg = String();
1283cdf0e10cSrcweir     pInst->nErr = 0L;
1284cdf0e10cSrcweir     pInst->nErl = 0;
1285cdf0e10cSrcweir     nError = 0L;
1286cdf0e10cSrcweir     SbxErrObject::getUnoErrObject()->Clear();
1287cdf0e10cSrcweir     bError = sal_False;
1288cdf0e10cSrcweir }
1289cdf0e10cSrcweir 
1290cdf0e10cSrcweir // UP verlassen
1291cdf0e10cSrcweir 
1292cdf0e10cSrcweir void SbiRuntime::StepLEAVE()
1293cdf0e10cSrcweir {
1294cdf0e10cSrcweir     bRun = sal_False;
1295cdf0e10cSrcweir         // If VBA and we are leaving an ErrorHandler then clear the error ( it's been processed )
1296cdf0e10cSrcweir     if ( bInError && pError )
1297cdf0e10cSrcweir         SbxErrObject::getUnoErrObject()->Clear();
1298cdf0e10cSrcweir }
1299cdf0e10cSrcweir 
1300cdf0e10cSrcweir void SbiRuntime::StepCHANNEL()      // TOS = Kanalnummer
1301cdf0e10cSrcweir {
1302cdf0e10cSrcweir     SbxVariableRef pChan = PopVar();
1303cdf0e10cSrcweir     short nChan = pChan->GetInteger();
1304cdf0e10cSrcweir     pIosys->SetChannel( nChan );
1305cdf0e10cSrcweir     Error( pIosys->GetError() );
1306cdf0e10cSrcweir }
1307cdf0e10cSrcweir 
1308cdf0e10cSrcweir void SbiRuntime::StepCHANNEL0()
1309cdf0e10cSrcweir {
1310cdf0e10cSrcweir     pIosys->ResetChannel();
1311cdf0e10cSrcweir }
1312cdf0e10cSrcweir 
1313cdf0e10cSrcweir void SbiRuntime::StepPRINT()        // print TOS
1314cdf0e10cSrcweir {
1315cdf0e10cSrcweir     SbxVariableRef p = PopVar();
1316cdf0e10cSrcweir     String s1 = p->GetString();
1317cdf0e10cSrcweir     String s;
1318cdf0e10cSrcweir     if( p->GetType() >= SbxINTEGER && p->GetType() <= SbxDOUBLE )
1319cdf0e10cSrcweir         s = ' ';    // ein Blank davor
1320cdf0e10cSrcweir     s += s1;
1321cdf0e10cSrcweir     ByteString aByteStr( s, gsl_getSystemTextEncoding() );
1322cdf0e10cSrcweir     pIosys->Write( aByteStr );
1323cdf0e10cSrcweir     Error( pIosys->GetError() );
1324cdf0e10cSrcweir }
1325cdf0e10cSrcweir 
1326cdf0e10cSrcweir void SbiRuntime::StepPRINTF()       // print TOS in field
1327cdf0e10cSrcweir {
1328cdf0e10cSrcweir     SbxVariableRef p = PopVar();
1329cdf0e10cSrcweir     String s1 = p->GetString();
1330cdf0e10cSrcweir     String s;
1331cdf0e10cSrcweir     if( p->GetType() >= SbxINTEGER && p->GetType() <= SbxDOUBLE )
1332cdf0e10cSrcweir         s = ' ';    // ein Blank davor
1333cdf0e10cSrcweir     s += s1;
1334cdf0e10cSrcweir     s.Expand( 14, ' ' );
1335cdf0e10cSrcweir     ByteString aByteStr( s, gsl_getSystemTextEncoding() );
1336cdf0e10cSrcweir     pIosys->Write( aByteStr );
1337cdf0e10cSrcweir     Error( pIosys->GetError() );
1338cdf0e10cSrcweir }
1339cdf0e10cSrcweir 
1340cdf0e10cSrcweir void SbiRuntime::StepWRITE()        // write TOS
1341cdf0e10cSrcweir {
1342cdf0e10cSrcweir     SbxVariableRef p = PopVar();
1343cdf0e10cSrcweir     // Muss der String gekapselt werden?
1344cdf0e10cSrcweir     char ch = 0;
1345cdf0e10cSrcweir     switch (p->GetType() )
1346cdf0e10cSrcweir     {
1347cdf0e10cSrcweir         case SbxSTRING: ch = '"'; break;
1348cdf0e10cSrcweir         case SbxCURRENCY:
1349cdf0e10cSrcweir         case SbxBOOL:
1350cdf0e10cSrcweir         case SbxDATE: ch = '#'; break;
1351cdf0e10cSrcweir         default: break;
1352cdf0e10cSrcweir     }
1353cdf0e10cSrcweir     String s;
1354cdf0e10cSrcweir     if( ch )
1355cdf0e10cSrcweir         s += ch;
1356cdf0e10cSrcweir     s += p->GetString();
1357cdf0e10cSrcweir     if( ch )
1358cdf0e10cSrcweir         s += ch;
1359cdf0e10cSrcweir     ByteString aByteStr( s, gsl_getSystemTextEncoding() );
1360cdf0e10cSrcweir     pIosys->Write( aByteStr );
1361cdf0e10cSrcweir     Error( pIosys->GetError() );
1362cdf0e10cSrcweir }
1363cdf0e10cSrcweir 
1364cdf0e10cSrcweir void SbiRuntime::StepRENAME()       // Rename Tos+1 to Tos
1365cdf0e10cSrcweir {
1366cdf0e10cSrcweir     SbxVariableRef pTos1 = PopVar();
1367cdf0e10cSrcweir     SbxVariableRef pTos  = PopVar();
1368cdf0e10cSrcweir     String aDest = pTos1->GetString();
1369cdf0e10cSrcweir     String aSource = pTos->GetString();
1370cdf0e10cSrcweir 
1371cdf0e10cSrcweir     // <-- UCB
1372cdf0e10cSrcweir     if( hasUno() )
1373cdf0e10cSrcweir     {
1374cdf0e10cSrcweir         implStepRenameUCB( aSource, aDest );
1375cdf0e10cSrcweir     }
1376cdf0e10cSrcweir     else
1377cdf0e10cSrcweir     // --> UCB
1378cdf0e10cSrcweir     {
1379cdf0e10cSrcweir #ifdef _OLD_FILE_IMPL
1380cdf0e10cSrcweir         DirEntry aSourceDirEntry( aSource );
1381cdf0e10cSrcweir         if( aSourceDirEntry.Exists() )
1382cdf0e10cSrcweir         {
1383cdf0e10cSrcweir             if( aSourceDirEntry.MoveTo( DirEntry(aDest) ) != FSYS_ERR_OK )
1384cdf0e10cSrcweir                 StarBASIC::Error( SbERR_PATH_NOT_FOUND );
1385cdf0e10cSrcweir         }
1386cdf0e10cSrcweir         else
1387cdf0e10cSrcweir                 StarBASIC::Error( SbERR_PATH_NOT_FOUND );
1388cdf0e10cSrcweir #else
1389cdf0e10cSrcweir         implStepRenameOSL( aSource, aDest );
1390cdf0e10cSrcweir #endif
1391cdf0e10cSrcweir     }
1392cdf0e10cSrcweir }
1393cdf0e10cSrcweir 
1394cdf0e10cSrcweir // TOS = Prompt
1395cdf0e10cSrcweir 
1396cdf0e10cSrcweir void SbiRuntime::StepPROMPT()
1397cdf0e10cSrcweir {
1398cdf0e10cSrcweir     SbxVariableRef p = PopVar();
1399cdf0e10cSrcweir     ByteString aStr( p->GetString(), gsl_getSystemTextEncoding() );
1400cdf0e10cSrcweir     pIosys->SetPrompt( aStr );
1401cdf0e10cSrcweir }
1402cdf0e10cSrcweir 
1403cdf0e10cSrcweir // Set Restart point
1404cdf0e10cSrcweir 
1405cdf0e10cSrcweir void SbiRuntime::StepRESTART()
1406cdf0e10cSrcweir {
1407cdf0e10cSrcweir     pRestart = pCode;
1408cdf0e10cSrcweir }
1409cdf0e10cSrcweir 
1410cdf0e10cSrcweir // Leerer Ausdruck auf Stack fuer fehlenden Parameter
1411cdf0e10cSrcweir 
1412cdf0e10cSrcweir void SbiRuntime::StepEMPTY()
1413cdf0e10cSrcweir {
1414cdf0e10cSrcweir     // #57915 Die Semantik von StepEMPTY() ist die Repraesentation eines fehlenden
1415cdf0e10cSrcweir     // Arguments. Dies wird in VB durch ein durch den Wert 448 (SbERR_NAMED_NOT_FOUND)
1416cdf0e10cSrcweir     // vom Typ Error repraesentiert. StepEmpty jetzt muesste besser StepMISSING()
1417cdf0e10cSrcweir     // heissen, aber der Name wird der Einfachkeit halber beibehalten.
1418cdf0e10cSrcweir     SbxVariableRef xVar = new SbxVariable( SbxVARIANT );
1419cdf0e10cSrcweir     xVar->PutErr( 448 );
1420cdf0e10cSrcweir     PushVar( xVar );
1421cdf0e10cSrcweir     // ALT: PushVar( new SbxVariable( SbxEMPTY ) );
1422cdf0e10cSrcweir }
1423cdf0e10cSrcweir 
1424cdf0e10cSrcweir // TOS = Fehlercode
1425cdf0e10cSrcweir 
1426cdf0e10cSrcweir void SbiRuntime::StepERROR()
1427cdf0e10cSrcweir {
1428cdf0e10cSrcweir     SbxVariableRef refCode = PopVar();
1429cdf0e10cSrcweir     sal_uInt16 n = refCode->GetUShort();
1430cdf0e10cSrcweir     SbError error = StarBASIC::GetSfxFromVBError( n );
1431cdf0e10cSrcweir     if ( bVBAEnabled )
1432cdf0e10cSrcweir         pInst->Error( error );
1433cdf0e10cSrcweir     else
1434cdf0e10cSrcweir         Error( error );
1435cdf0e10cSrcweir }
1436cdf0e10cSrcweir 
1437