xref: /aoo4110/main/sc/source/filter/excel/xilink.cxx (revision b1cdbd2c)
1*b1cdbd2cSJim Jagielski /**************************************************************
2*b1cdbd2cSJim Jagielski  *
3*b1cdbd2cSJim Jagielski  * Licensed to the Apache Software Foundation (ASF) under one
4*b1cdbd2cSJim Jagielski  * or more contributor license agreements.  See the NOTICE file
5*b1cdbd2cSJim Jagielski  * distributed with this work for additional information
6*b1cdbd2cSJim Jagielski  * regarding copyright ownership.  The ASF licenses this file
7*b1cdbd2cSJim Jagielski  * to you under the Apache License, Version 2.0 (the
8*b1cdbd2cSJim Jagielski  * "License"); you may not use this file except in compliance
9*b1cdbd2cSJim Jagielski  * with the License.  You may obtain a copy of the License at
10*b1cdbd2cSJim Jagielski  *
11*b1cdbd2cSJim Jagielski  *   http://www.apache.org/licenses/LICENSE-2.0
12*b1cdbd2cSJim Jagielski  *
13*b1cdbd2cSJim Jagielski  * Unless required by applicable law or agreed to in writing,
14*b1cdbd2cSJim Jagielski  * software distributed under the License is distributed on an
15*b1cdbd2cSJim Jagielski  * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
16*b1cdbd2cSJim Jagielski  * KIND, either express or implied.  See the License for the
17*b1cdbd2cSJim Jagielski  * specific language governing permissions and limitations
18*b1cdbd2cSJim Jagielski  * under the License.
19*b1cdbd2cSJim Jagielski  *
20*b1cdbd2cSJim Jagielski  *************************************************************/
21*b1cdbd2cSJim Jagielski 
22*b1cdbd2cSJim Jagielski 
23*b1cdbd2cSJim Jagielski 
24*b1cdbd2cSJim Jagielski // MARKER(update_precomp.py): autogen include statement, do not remove
25*b1cdbd2cSJim Jagielski #include "precompiled_sc.hxx"
26*b1cdbd2cSJim Jagielski #include "xilink.hxx"
27*b1cdbd2cSJim Jagielski #include "document.hxx"
28*b1cdbd2cSJim Jagielski #include "cell.hxx"
29*b1cdbd2cSJim Jagielski #include "scextopt.hxx"
30*b1cdbd2cSJim Jagielski #include "tablink.hxx"
31*b1cdbd2cSJim Jagielski #include "xistream.hxx"
32*b1cdbd2cSJim Jagielski #include "xihelper.hxx"
33*b1cdbd2cSJim Jagielski #include "xiname.hxx"
34*b1cdbd2cSJim Jagielski #include "excform.hxx"
35*b1cdbd2cSJim Jagielski #include "tokenarray.hxx"
36*b1cdbd2cSJim Jagielski #include "externalrefmgr.hxx"
37*b1cdbd2cSJim Jagielski 
38*b1cdbd2cSJim Jagielski #include <vector>
39*b1cdbd2cSJim Jagielski 
40*b1cdbd2cSJim Jagielski using ::std::vector;
41*b1cdbd2cSJim Jagielski 
42*b1cdbd2cSJim Jagielski // ============================================================================
43*b1cdbd2cSJim Jagielski // *** Helper classes ***
44*b1cdbd2cSJim Jagielski // ============================================================================
45*b1cdbd2cSJim Jagielski 
46*b1cdbd2cSJim Jagielski // Cached external cells ======================================================
47*b1cdbd2cSJim Jagielski 
48*b1cdbd2cSJim Jagielski /** Contains the address and value of an external referenced cell. */
49*b1cdbd2cSJim Jagielski class XclImpCrn : public XclImpCachedValue
50*b1cdbd2cSJim Jagielski {
51*b1cdbd2cSJim Jagielski public:
52*b1cdbd2cSJim Jagielski     /** Reads a cached value and stores it with its cell address. */
53*b1cdbd2cSJim Jagielski     explicit            XclImpCrn( XclImpStream& rStrm, const XclAddress& rXclPos );
54*b1cdbd2cSJim Jagielski 
55*b1cdbd2cSJim Jagielski     const XclAddress&   GetAddress() const;
56*b1cdbd2cSJim Jagielski 
57*b1cdbd2cSJim Jagielski private:
58*b1cdbd2cSJim Jagielski     XclAddress          maXclPos;       /// Excel position of the cached cell.
59*b1cdbd2cSJim Jagielski };
60*b1cdbd2cSJim Jagielski 
61*b1cdbd2cSJim Jagielski // Sheet in an external document ==============================================
62*b1cdbd2cSJim Jagielski 
63*b1cdbd2cSJim Jagielski /** Contains the name and sheet index of one sheet in an external document. */
64*b1cdbd2cSJim Jagielski class XclImpSupbookTab
65*b1cdbd2cSJim Jagielski {
66*b1cdbd2cSJim Jagielski public:
67*b1cdbd2cSJim Jagielski     /** Stores the sheet name and marks the sheet index as invalid.
68*b1cdbd2cSJim Jagielski         The sheet index is set while creating the Calc sheet with CreateTable(). */
69*b1cdbd2cSJim Jagielski     explicit            XclImpSupbookTab( const String& rTabName );
70*b1cdbd2cSJim Jagielski                         ~XclImpSupbookTab();
71*b1cdbd2cSJim Jagielski 
GetTabName() const72*b1cdbd2cSJim Jagielski     inline const String& GetTabName() const { return maTabName; }
73*b1cdbd2cSJim Jagielski 
74*b1cdbd2cSJim Jagielski     /** Reads a CRN record (external referenced cell) at the specified address. */
75*b1cdbd2cSJim Jagielski     void                ReadCrn( XclImpStream& rStrm, const XclAddress& rXclPos );
76*b1cdbd2cSJim Jagielski 
77*b1cdbd2cSJim Jagielski     void                LoadCachedValues(ScExternalRefCache::TableTypeRef pCacheTable);
78*b1cdbd2cSJim Jagielski 
79*b1cdbd2cSJim Jagielski private:
80*b1cdbd2cSJim Jagielski     typedef ScfDelList< XclImpCrn > XclImpCrnList;
81*b1cdbd2cSJim Jagielski 
82*b1cdbd2cSJim Jagielski     XclImpCrnList       maCrnList;      /// List of CRN records (cached cell values).
83*b1cdbd2cSJim Jagielski     String              maTabName;      /// Name of the external sheet.
84*b1cdbd2cSJim Jagielski     SCTAB               mnScTab;        /// New sheet index in Calc document.
85*b1cdbd2cSJim Jagielski };
86*b1cdbd2cSJim Jagielski 
87*b1cdbd2cSJim Jagielski // External document (SUPBOOK) ================================================
88*b1cdbd2cSJim Jagielski 
89*b1cdbd2cSJim Jagielski /** This class represents an external linked document (record SUPBOOK).
90*b1cdbd2cSJim Jagielski     @descr  Contains a list of all referenced sheets in the document. */
91*b1cdbd2cSJim Jagielski class XclImpSupbook : protected XclImpRoot
92*b1cdbd2cSJim Jagielski {
93*b1cdbd2cSJim Jagielski public:
94*b1cdbd2cSJim Jagielski     /** Reads the SUPBOOK record from stream. */
95*b1cdbd2cSJim Jagielski     explicit            XclImpSupbook( XclImpStream& rStrm );
96*b1cdbd2cSJim Jagielski 
97*b1cdbd2cSJim Jagielski     /** Reads an XCT record (count of following CRNs and current sheet). */
98*b1cdbd2cSJim Jagielski     void                ReadXct( XclImpStream& rStrm );
99*b1cdbd2cSJim Jagielski     /** Reads a CRN record (external referenced cell). */
100*b1cdbd2cSJim Jagielski     void                ReadCrn( XclImpStream& rStrm );
101*b1cdbd2cSJim Jagielski     /** Reads an EXTERNNAME record. */
102*b1cdbd2cSJim Jagielski     void                ReadExternname( XclImpStream& rStrm, ExcelToSc* pFormulaConv = NULL );
103*b1cdbd2cSJim Jagielski 
104*b1cdbd2cSJim Jagielski     /** Returns the SUPBOOK record type. */
GetType() const105*b1cdbd2cSJim Jagielski     inline XclSupbookType GetType() const { return meType; }
106*b1cdbd2cSJim Jagielski 
107*b1cdbd2cSJim Jagielski     /** Returns the URL of the external document. */
GetXclUrl() const108*b1cdbd2cSJim Jagielski     inline const String& GetXclUrl() const { return maXclUrl; }
109*b1cdbd2cSJim Jagielski 
110*b1cdbd2cSJim Jagielski     /** Returns the external name specified by an index from the Excel document (one-based). */
111*b1cdbd2cSJim Jagielski     const XclImpExtName* GetExternName( sal_uInt16 nXclIndex ) const;
112*b1cdbd2cSJim Jagielski     /** Tries to decode the URL to OLE or DDE link components.
113*b1cdbd2cSJim Jagielski         @descr  For DDE links: Decodes to application name and topic.
114*b1cdbd2cSJim Jagielski         For OLE object links: Decodes to class name and document URL.
115*b1cdbd2cSJim Jagielski         @return  true = decoding was successful, returned strings are valid (not empty). */
116*b1cdbd2cSJim Jagielski     bool                GetLinkData( String& rApplic, String& rDoc ) const;
117*b1cdbd2cSJim Jagielski     /** Returns the specified macro name (1-based) or an empty string on error. */
118*b1cdbd2cSJim Jagielski     const String&       GetMacroName( sal_uInt16 nXclNameIdx ) const;
119*b1cdbd2cSJim Jagielski 
120*b1cdbd2cSJim Jagielski     const String&       GetTabName( sal_uInt16 nXtiTab ) const;
121*b1cdbd2cSJim Jagielski 
122*b1cdbd2cSJim Jagielski     sal_uInt16          GetTabCount() const;
123*b1cdbd2cSJim Jagielski 
124*b1cdbd2cSJim Jagielski     void                LoadCachedValues();
125*b1cdbd2cSJim Jagielski 
126*b1cdbd2cSJim Jagielski private:
127*b1cdbd2cSJim Jagielski     typedef ScfDelList< XclImpSupbookTab >  XclImpSupbookTabList;
128*b1cdbd2cSJim Jagielski     typedef ScfDelList< XclImpExtName >     XclImpExtNameList;
129*b1cdbd2cSJim Jagielski 
130*b1cdbd2cSJim Jagielski     XclImpSupbookTabList maSupbTabList;     /// All sheet names of the document.
131*b1cdbd2cSJim Jagielski     XclImpExtNameList   maExtNameList;      /// All external names of the document.
132*b1cdbd2cSJim Jagielski     String              maXclUrl;           /// URL of the external document (Excel mode).
133*b1cdbd2cSJim Jagielski     String              maFilterName;       /// Detected filer name.
134*b1cdbd2cSJim Jagielski     String              maFilterOpt;        /// Detected filer options.
135*b1cdbd2cSJim Jagielski     XclSupbookType      meType;             /// Type of the supbook record.
136*b1cdbd2cSJim Jagielski     sal_uInt16          mnSBTab;            /// Current Excel sheet index from SUPBOOK for XCT/CRN records.
137*b1cdbd2cSJim Jagielski };
138*b1cdbd2cSJim Jagielski 
139*b1cdbd2cSJim Jagielski // Import link manager ========================================================
140*b1cdbd2cSJim Jagielski 
141*b1cdbd2cSJim Jagielski /** Contains the SUPBOOK index and sheet indexes of an external link.
142*b1cdbd2cSJim Jagielski     @descr  It is possible to enter a formula like =SUM(Sheet1:Sheet3!A1),
143*b1cdbd2cSJim Jagielski     therefore here occurs a sheet range. */
144*b1cdbd2cSJim Jagielski struct XclImpXti
145*b1cdbd2cSJim Jagielski {
146*b1cdbd2cSJim Jagielski     sal_uInt16          mnSupbook;      /// Index to SUPBOOK record.
147*b1cdbd2cSJim Jagielski     sal_uInt16          mnSBTabFirst;   /// Index to the first sheet of the range in the SUPBOOK.
148*b1cdbd2cSJim Jagielski     sal_uInt16          mnSBTabLast;    /// Index to the last sheet of the range in the SUPBOOK.
XclImpXtiXclImpXti149*b1cdbd2cSJim Jagielski     inline explicit     XclImpXti() : mnSupbook( SAL_MAX_UINT16 ), mnSBTabFirst( SAL_MAX_UINT16 ), mnSBTabLast( SAL_MAX_UINT16 ) {}
150*b1cdbd2cSJim Jagielski };
151*b1cdbd2cSJim Jagielski 
operator >>(XclImpStream & rStrm,XclImpXti & rXti)152*b1cdbd2cSJim Jagielski inline XclImpStream& operator>>( XclImpStream& rStrm, XclImpXti& rXti )
153*b1cdbd2cSJim Jagielski {
154*b1cdbd2cSJim Jagielski     return rStrm >> rXti.mnSupbook >> rXti.mnSBTabFirst >> rXti.mnSBTabLast;
155*b1cdbd2cSJim Jagielski }
156*b1cdbd2cSJim Jagielski 
157*b1cdbd2cSJim Jagielski // ----------------------------------------------------------------------------
158*b1cdbd2cSJim Jagielski 
159*b1cdbd2cSJim Jagielski /** Implementation of the link manager. */
160*b1cdbd2cSJim Jagielski class XclImpLinkManagerImpl : protected XclImpRoot
161*b1cdbd2cSJim Jagielski {
162*b1cdbd2cSJim Jagielski public:
163*b1cdbd2cSJim Jagielski     explicit            XclImpLinkManagerImpl( const XclImpRoot& rRoot );
164*b1cdbd2cSJim Jagielski 
165*b1cdbd2cSJim Jagielski     /** Reads the EXTERNSHEET record. */
166*b1cdbd2cSJim Jagielski     void                ReadExternsheet( XclImpStream& rStrm );
167*b1cdbd2cSJim Jagielski     /** Reads a SUPBOOK record. */
168*b1cdbd2cSJim Jagielski     void                ReadSupbook( XclImpStream& rStrm );
169*b1cdbd2cSJim Jagielski     /** Reads an XCT record and appends it to the current SUPBOOK. */
170*b1cdbd2cSJim Jagielski     void                ReadXct( XclImpStream& rStrm );
171*b1cdbd2cSJim Jagielski     /** Reads a CRN record and appends it to the current SUPBOOK. */
172*b1cdbd2cSJim Jagielski     void                ReadCrn( XclImpStream& rStrm );
173*b1cdbd2cSJim Jagielski     /** Reads an EXTERNNAME record and appends it to the current SUPBOOK. */
174*b1cdbd2cSJim Jagielski     void                ReadExternname( XclImpStream& rStrm, ExcelToSc* pFormulaConv = NULL );
175*b1cdbd2cSJim Jagielski 
176*b1cdbd2cSJim Jagielski     /** Returns true, if the specified XTI entry contains an internal reference. */
177*b1cdbd2cSJim Jagielski     bool                IsSelfRef( sal_uInt16 nXtiIndex ) const;
178*b1cdbd2cSJim Jagielski     /** Returns the Calc sheet index range of the specified XTI entry.
179*b1cdbd2cSJim Jagielski         @return  true = XTI data found, returned sheet index range is valid. */
180*b1cdbd2cSJim Jagielski     bool                GetScTabRange(
181*b1cdbd2cSJim Jagielski                             SCTAB& rnFirstScTab, SCTAB& rnLastScTab,
182*b1cdbd2cSJim Jagielski                             sal_uInt16 nXtiIndex ) const;
183*b1cdbd2cSJim Jagielski     /** Returns the specified external name or 0 on error. */
184*b1cdbd2cSJim Jagielski     const XclImpExtName* GetExternName( sal_uInt16 nXtiIndex, sal_uInt16 nExtName ) const;
185*b1cdbd2cSJim Jagielski 
186*b1cdbd2cSJim Jagielski     /** Returns the absolute file URL of a supporting workbook specified by
187*b1cdbd2cSJim Jagielski         the index. */
188*b1cdbd2cSJim Jagielski     const String*       GetSupbookUrl( sal_uInt16 nXtiIndex ) const;
189*b1cdbd2cSJim Jagielski 
190*b1cdbd2cSJim Jagielski     const String&       GetSupbookTabName( sal_uInt16 nXti, sal_uInt16 nXtiTab ) const;
191*b1cdbd2cSJim Jagielski 
192*b1cdbd2cSJim Jagielski     /** Tries to decode the URL of the specified XTI entry to OLE or DDE link components.
193*b1cdbd2cSJim Jagielski         @descr  For DDE links: Decodes to application name and topic.
194*b1cdbd2cSJim Jagielski         For OLE object links: Decodes to class name and document URL.
195*b1cdbd2cSJim Jagielski         @return  true = decoding was successful, returned strings are valid (not empty). */
196*b1cdbd2cSJim Jagielski     bool                GetLinkData( String& rApplic, String& rTopic, sal_uInt16 nXtiIndex ) const;
197*b1cdbd2cSJim Jagielski     /** Returns the specified macro name or an empty string on error. */
198*b1cdbd2cSJim Jagielski     const String&       GetMacroName( sal_uInt16 nExtSheet, sal_uInt16 nExtName ) const;
199*b1cdbd2cSJim Jagielski 
200*b1cdbd2cSJim Jagielski private:
201*b1cdbd2cSJim Jagielski     /** Returns the specified XTI (link entry from BIFF8 EXTERNSHEET record). */
202*b1cdbd2cSJim Jagielski     const XclImpXti*    GetXti( sal_uInt16 nXtiIndex ) const;
203*b1cdbd2cSJim Jagielski     /** Returns the specified SUPBOOK (external document). */
204*b1cdbd2cSJim Jagielski     const XclImpSupbook* GetSupbook( sal_uInt16 nXtiIndex ) const;
205*b1cdbd2cSJim Jagielski //UNUSED2009-05 /** Returns the SUPBOOK (external workbook) specified by its URL. */
206*b1cdbd2cSJim Jagielski //UNUSED2009-05 const XclImpSupbook* GetSupbook( const String& rUrl ) const;
207*b1cdbd2cSJim Jagielski 
208*b1cdbd2cSJim Jagielski     void                LoadCachedValues();
209*b1cdbd2cSJim Jagielski 
210*b1cdbd2cSJim Jagielski //UNUSED2009-05 /** Finds the largest range of sheet indexes in a SUPBOOK after a start sheet index.
211*b1cdbd2cSJim Jagielski //UNUSED2009-05     @param rnSBTabFirst  (out-param) The first sheet index of the range in SUPBOOK is returned here.
212*b1cdbd2cSJim Jagielski //UNUSED2009-05     @param rnSBTabLast  (out-param) The last sheet index of the range in SUPBOOK is returned here (inclusive).
213*b1cdbd2cSJim Jagielski //UNUSED2009-05     @param nSupbook  The list index of the SUPBOOK.
214*b1cdbd2cSJim Jagielski //UNUSED2009-05     @param nSBTabStart  The first allowed sheet index. Sheet ranges with an earlier start index are ignored.
215*b1cdbd2cSJim Jagielski //UNUSED2009-05     @return  true = the return values are valid; false = nothing found. */
216*b1cdbd2cSJim Jagielski //UNUSED2009-05 bool                FindNextTabRange(
217*b1cdbd2cSJim Jagielski //UNUSED2009-05                         sal_uInt16& rnSBTabFirst, sal_uInt16& rnSBTabLast,
218*b1cdbd2cSJim Jagielski //UNUSED2009-05                         sal_uInt16 nSupbook, sal_uInt16 nSBTabStart ) const;
219*b1cdbd2cSJim Jagielski 
220*b1cdbd2cSJim Jagielski private:
221*b1cdbd2cSJim Jagielski     typedef ::std::vector< XclImpXti >  XclImpXtiVector;
222*b1cdbd2cSJim Jagielski     typedef ScfDelList< XclImpSupbook > XclImpSupbookList;
223*b1cdbd2cSJim Jagielski 
224*b1cdbd2cSJim Jagielski     XclImpXtiVector     maXtiList;          /// List of all XTI structures.
225*b1cdbd2cSJim Jagielski     XclImpSupbookList   maSupbookList;      /// List of external documents.
226*b1cdbd2cSJim Jagielski     bool                mbCreated;          /// true = Calc sheets already created.
227*b1cdbd2cSJim Jagielski };
228*b1cdbd2cSJim Jagielski 
229*b1cdbd2cSJim Jagielski // ============================================================================
230*b1cdbd2cSJim Jagielski // *** Implementation ***
231*b1cdbd2cSJim Jagielski // ============================================================================
232*b1cdbd2cSJim Jagielski 
233*b1cdbd2cSJim Jagielski // Excel sheet indexes ========================================================
234*b1cdbd2cSJim Jagielski 
235*b1cdbd2cSJim Jagielski // original Excel sheet names -------------------------------------------------
236*b1cdbd2cSJim Jagielski 
AppendXclTabName(const String & rXclTabName,SCTAB nScTab)237*b1cdbd2cSJim Jagielski void XclImpTabInfo::AppendXclTabName( const String& rXclTabName, SCTAB nScTab )
238*b1cdbd2cSJim Jagielski {
239*b1cdbd2cSJim Jagielski     maTabNames[ rXclTabName ] = nScTab;
240*b1cdbd2cSJim Jagielski }
241*b1cdbd2cSJim Jagielski 
InsertScTab(SCTAB nScTab)242*b1cdbd2cSJim Jagielski void XclImpTabInfo::InsertScTab( SCTAB nScTab )
243*b1cdbd2cSJim Jagielski {
244*b1cdbd2cSJim Jagielski     for( XclTabNameMap::iterator aIt = maTabNames.begin(), aEnd = maTabNames.end(); aIt != aEnd; ++aIt )
245*b1cdbd2cSJim Jagielski         if( aIt->second >= nScTab )
246*b1cdbd2cSJim Jagielski             ++aIt->second;
247*b1cdbd2cSJim Jagielski }
248*b1cdbd2cSJim Jagielski 
GetScTabFromXclName(const String & rXclTabName) const249*b1cdbd2cSJim Jagielski SCTAB XclImpTabInfo::GetScTabFromXclName( const String& rXclTabName ) const
250*b1cdbd2cSJim Jagielski {
251*b1cdbd2cSJim Jagielski     XclTabNameMap::const_iterator aIt = maTabNames.find( rXclTabName );
252*b1cdbd2cSJim Jagielski     return (aIt != maTabNames.end()) ? aIt->second : SCTAB_INVALID;
253*b1cdbd2cSJim Jagielski }
254*b1cdbd2cSJim Jagielski 
255*b1cdbd2cSJim Jagielski // record creation order - TABID record ---------------------------------------
256*b1cdbd2cSJim Jagielski 
ReadTabid(XclImpStream & rStrm)257*b1cdbd2cSJim Jagielski void XclImpTabInfo::ReadTabid( XclImpStream& rStrm )
258*b1cdbd2cSJim Jagielski {
259*b1cdbd2cSJim Jagielski     DBG_ASSERT_BIFF( rStrm.GetRoot().GetBiff() == EXC_BIFF8 );
260*b1cdbd2cSJim Jagielski     if( rStrm.GetRoot().GetBiff() == EXC_BIFF8 )
261*b1cdbd2cSJim Jagielski     {
262*b1cdbd2cSJim Jagielski         rStrm.EnableDecryption();
263*b1cdbd2cSJim Jagielski         sal_Size nReadCount = rStrm.GetRecLeft() / 2;
264*b1cdbd2cSJim Jagielski         DBG_ASSERT( nReadCount <= 0xFFFF, "XclImpTabInfo::ReadTabid - record too long" );
265*b1cdbd2cSJim Jagielski         maTabIdVec.clear();
266*b1cdbd2cSJim Jagielski         maTabIdVec.reserve( nReadCount );
267*b1cdbd2cSJim Jagielski         for( sal_Size nIndex = 0; rStrm.IsValid() && (nIndex < nReadCount); ++nIndex )
268*b1cdbd2cSJim Jagielski             // #93471# zero index is not allowed in BIFF8, but it seems that it occurs in real life
269*b1cdbd2cSJim Jagielski             maTabIdVec.push_back( rStrm.ReaduInt16() );
270*b1cdbd2cSJim Jagielski     }
271*b1cdbd2cSJim Jagielski }
272*b1cdbd2cSJim Jagielski 
GetCurrentIndex(sal_uInt16 nCreatedId,sal_uInt16 nMaxTabId) const273*b1cdbd2cSJim Jagielski sal_uInt16 XclImpTabInfo::GetCurrentIndex( sal_uInt16 nCreatedId, sal_uInt16 nMaxTabId ) const
274*b1cdbd2cSJim Jagielski {
275*b1cdbd2cSJim Jagielski     sal_uInt16 nReturn = 0;
276*b1cdbd2cSJim Jagielski     for( ScfUInt16Vec::const_iterator aIt = maTabIdVec.begin(), aEnd = maTabIdVec.end(); aIt != aEnd; ++aIt )
277*b1cdbd2cSJim Jagielski     {
278*b1cdbd2cSJim Jagielski         sal_uInt16 nValue = *aIt;
279*b1cdbd2cSJim Jagielski         if( nValue == nCreatedId )
280*b1cdbd2cSJim Jagielski             return nReturn;
281*b1cdbd2cSJim Jagielski         if( nValue <= nMaxTabId )
282*b1cdbd2cSJim Jagielski             ++nReturn;
283*b1cdbd2cSJim Jagielski     }
284*b1cdbd2cSJim Jagielski     return 0;
285*b1cdbd2cSJim Jagielski }
286*b1cdbd2cSJim Jagielski 
287*b1cdbd2cSJim Jagielski // External names =============================================================
288*b1cdbd2cSJim Jagielski 
XclImpExtName(const XclImpSupbook & rSupbook,XclImpStream & rStrm,XclSupbookType eSubType,ExcelToSc * pFormulaConv)289*b1cdbd2cSJim Jagielski XclImpExtName::XclImpExtName( const XclImpSupbook& rSupbook, XclImpStream& rStrm, XclSupbookType eSubType, ExcelToSc* pFormulaConv )
290*b1cdbd2cSJim Jagielski {
291*b1cdbd2cSJim Jagielski     sal_uInt16 nFlags;
292*b1cdbd2cSJim Jagielski     sal_uInt8 nLen;
293*b1cdbd2cSJim Jagielski 
294*b1cdbd2cSJim Jagielski     rStrm >> nFlags >> mnStorageId >> nLen ;
295*b1cdbd2cSJim Jagielski     maName = rStrm.ReadUniString( nLen );
296*b1cdbd2cSJim Jagielski     if( ::get_flag( nFlags, EXC_EXTN_BUILTIN ) || !::get_flag( nFlags, EXC_EXTN_OLE_OR_DDE ) )
297*b1cdbd2cSJim Jagielski     {
298*b1cdbd2cSJim Jagielski         if( eSubType == EXC_SBTYPE_ADDIN )
299*b1cdbd2cSJim Jagielski         {
300*b1cdbd2cSJim Jagielski             meType = xlExtAddIn;
301*b1cdbd2cSJim Jagielski             maName = rStrm.GetRoot().GetScAddInName( maName );
302*b1cdbd2cSJim Jagielski         }
303*b1cdbd2cSJim Jagielski         else if ( (eSubType == EXC_SBTYPE_EUROTOOL) &&
304*b1cdbd2cSJim Jagielski                 maName.EqualsIgnoreCaseAscii( "EUROCONVERT" ) )
305*b1cdbd2cSJim Jagielski             meType = xlExtEuroConvert;
306*b1cdbd2cSJim Jagielski         else
307*b1cdbd2cSJim Jagielski         {
308*b1cdbd2cSJim Jagielski             meType = xlExtName;
309*b1cdbd2cSJim Jagielski             ScfTools::ConvertToScDefinedName( maName );
310*b1cdbd2cSJim Jagielski         }
311*b1cdbd2cSJim Jagielski     }
312*b1cdbd2cSJim Jagielski     else
313*b1cdbd2cSJim Jagielski     {
314*b1cdbd2cSJim Jagielski         meType = ::get_flagvalue( nFlags, EXC_EXTN_OLE, xlExtOLE, xlExtDDE );
315*b1cdbd2cSJim Jagielski     }
316*b1cdbd2cSJim Jagielski 
317*b1cdbd2cSJim Jagielski     if( (meType == xlExtDDE) && (rStrm.GetRecLeft() > 1) )
318*b1cdbd2cSJim Jagielski         mxDdeMatrix.reset( new XclImpCachedMatrix( rStrm ) );
319*b1cdbd2cSJim Jagielski 
320*b1cdbd2cSJim Jagielski     if (meType == xlExtName)
321*b1cdbd2cSJim Jagielski     {
322*b1cdbd2cSJim Jagielski         // TODO: For now, only global external names are supported.  In future
323*b1cdbd2cSJim Jagielski         // we should extend this to supporting per-sheet external names.
324*b1cdbd2cSJim Jagielski         if (mnStorageId == 0)
325*b1cdbd2cSJim Jagielski         {
326*b1cdbd2cSJim Jagielski             if (pFormulaConv)
327*b1cdbd2cSJim Jagielski             {
328*b1cdbd2cSJim Jagielski                 const ScTokenArray* pArray = NULL;
329*b1cdbd2cSJim Jagielski                 sal_uInt16 nFmlaLen;
330*b1cdbd2cSJim Jagielski                 rStrm >> nFmlaLen;
331*b1cdbd2cSJim Jagielski                 vector<String> aTabNames;
332*b1cdbd2cSJim Jagielski                 sal_uInt16 nCount = rSupbook.GetTabCount();
333*b1cdbd2cSJim Jagielski                 aTabNames.reserve(nCount);
334*b1cdbd2cSJim Jagielski                 for (sal_uInt16 i = 0; i < nCount; ++i)
335*b1cdbd2cSJim Jagielski                     aTabNames.push_back(rSupbook.GetTabName(i));
336*b1cdbd2cSJim Jagielski 
337*b1cdbd2cSJim Jagielski                 pFormulaConv->ConvertExternName(pArray, rStrm, nFmlaLen, rSupbook.GetXclUrl(), aTabNames);
338*b1cdbd2cSJim Jagielski                 if (pArray)
339*b1cdbd2cSJim Jagielski                     mxArray.reset(pArray->Clone());
340*b1cdbd2cSJim Jagielski             }
341*b1cdbd2cSJim Jagielski         }
342*b1cdbd2cSJim Jagielski     }
343*b1cdbd2cSJim Jagielski }
344*b1cdbd2cSJim Jagielski 
~XclImpExtName()345*b1cdbd2cSJim Jagielski XclImpExtName::~XclImpExtName()
346*b1cdbd2cSJim Jagielski {
347*b1cdbd2cSJim Jagielski }
348*b1cdbd2cSJim Jagielski 
CreateDdeData(ScDocument & rDoc,const String & rApplic,const String & rTopic) const349*b1cdbd2cSJim Jagielski void XclImpExtName::CreateDdeData( ScDocument& rDoc, const String& rApplic, const String& rTopic ) const
350*b1cdbd2cSJim Jagielski {
351*b1cdbd2cSJim Jagielski     ScMatrixRef xResults;
352*b1cdbd2cSJim Jagielski     if( mxDdeMatrix.get() )
353*b1cdbd2cSJim Jagielski         xResults = mxDdeMatrix->CreateScMatrix();
354*b1cdbd2cSJim Jagielski     rDoc.CreateDdeLink( rApplic, rTopic, maName, SC_DDE_DEFAULT, xResults );
355*b1cdbd2cSJim Jagielski }
356*b1cdbd2cSJim Jagielski 
CreateExtNameData(ScDocument & rDoc,sal_uInt16 nFileId) const357*b1cdbd2cSJim Jagielski void XclImpExtName::CreateExtNameData( ScDocument& rDoc, sal_uInt16 nFileId ) const
358*b1cdbd2cSJim Jagielski {
359*b1cdbd2cSJim Jagielski     if (!mxArray.get())
360*b1cdbd2cSJim Jagielski         return;
361*b1cdbd2cSJim Jagielski 
362*b1cdbd2cSJim Jagielski     ScExternalRefManager* pRefMgr = rDoc.GetExternalRefManager();
363*b1cdbd2cSJim Jagielski     pRefMgr->storeRangeNameTokens(nFileId, maName, *mxArray);
364*b1cdbd2cSJim Jagielski }
365*b1cdbd2cSJim Jagielski 
HasFormulaTokens() const366*b1cdbd2cSJim Jagielski bool XclImpExtName::HasFormulaTokens() const
367*b1cdbd2cSJim Jagielski {
368*b1cdbd2cSJim Jagielski     return (mxArray.get() != NULL);
369*b1cdbd2cSJim Jagielski }
370*b1cdbd2cSJim Jagielski 
371*b1cdbd2cSJim Jagielski // Cached external cells ======================================================
372*b1cdbd2cSJim Jagielski 
XclImpCrn(XclImpStream & rStrm,const XclAddress & rXclPos)373*b1cdbd2cSJim Jagielski XclImpCrn::XclImpCrn( XclImpStream& rStrm, const XclAddress& rXclPos ) :
374*b1cdbd2cSJim Jagielski     XclImpCachedValue( rStrm ),
375*b1cdbd2cSJim Jagielski     maXclPos( rXclPos )
376*b1cdbd2cSJim Jagielski {
377*b1cdbd2cSJim Jagielski }
378*b1cdbd2cSJim Jagielski 
GetAddress() const379*b1cdbd2cSJim Jagielski const XclAddress& XclImpCrn::GetAddress() const
380*b1cdbd2cSJim Jagielski {
381*b1cdbd2cSJim Jagielski     return maXclPos;
382*b1cdbd2cSJim Jagielski }
383*b1cdbd2cSJim Jagielski 
384*b1cdbd2cSJim Jagielski // Sheet in an external document ==============================================
385*b1cdbd2cSJim Jagielski 
XclImpSupbookTab(const String & rTabName)386*b1cdbd2cSJim Jagielski XclImpSupbookTab::XclImpSupbookTab( const String& rTabName ) :
387*b1cdbd2cSJim Jagielski     maTabName( rTabName ),
388*b1cdbd2cSJim Jagielski     mnScTab( SCTAB_INVALID )
389*b1cdbd2cSJim Jagielski {
390*b1cdbd2cSJim Jagielski }
391*b1cdbd2cSJim Jagielski 
~XclImpSupbookTab()392*b1cdbd2cSJim Jagielski XclImpSupbookTab::~XclImpSupbookTab()
393*b1cdbd2cSJim Jagielski {
394*b1cdbd2cSJim Jagielski }
395*b1cdbd2cSJim Jagielski 
ReadCrn(XclImpStream & rStrm,const XclAddress & rXclPos)396*b1cdbd2cSJim Jagielski void XclImpSupbookTab::ReadCrn( XclImpStream& rStrm, const XclAddress& rXclPos )
397*b1cdbd2cSJim Jagielski {
398*b1cdbd2cSJim Jagielski     maCrnList.Append( new XclImpCrn( rStrm, rXclPos ) );
399*b1cdbd2cSJim Jagielski }
400*b1cdbd2cSJim Jagielski 
LoadCachedValues(ScExternalRefCache::TableTypeRef pCacheTable)401*b1cdbd2cSJim Jagielski void XclImpSupbookTab::LoadCachedValues(ScExternalRefCache::TableTypeRef pCacheTable)
402*b1cdbd2cSJim Jagielski {
403*b1cdbd2cSJim Jagielski     if (maCrnList.Empty())
404*b1cdbd2cSJim Jagielski         return;
405*b1cdbd2cSJim Jagielski 
406*b1cdbd2cSJim Jagielski     for (XclImpCrn* p = maCrnList.First(); p; p = maCrnList.Next())
407*b1cdbd2cSJim Jagielski     {
408*b1cdbd2cSJim Jagielski         const XclAddress& rAddr = p->GetAddress();
409*b1cdbd2cSJim Jagielski         switch (p->GetType())
410*b1cdbd2cSJim Jagielski         {
411*b1cdbd2cSJim Jagielski             case EXC_CACHEDVAL_BOOL:
412*b1cdbd2cSJim Jagielski             {
413*b1cdbd2cSJim Jagielski                 bool b = p->GetBool();
414*b1cdbd2cSJim Jagielski                 ScExternalRefCache::TokenRef pToken(new formula::FormulaDoubleToken(b ? 1.0 : 0.0));
415*b1cdbd2cSJim Jagielski                 pCacheTable->setCell(rAddr.mnCol, rAddr.mnRow, pToken);
416*b1cdbd2cSJim Jagielski             }
417*b1cdbd2cSJim Jagielski             break;
418*b1cdbd2cSJim Jagielski             case EXC_CACHEDVAL_DOUBLE:
419*b1cdbd2cSJim Jagielski             {
420*b1cdbd2cSJim Jagielski                 double f = p->GetValue();
421*b1cdbd2cSJim Jagielski                 ScExternalRefCache::TokenRef pToken(new formula::FormulaDoubleToken(f));
422*b1cdbd2cSJim Jagielski                 pCacheTable->setCell(rAddr.mnCol, rAddr.mnRow, pToken);
423*b1cdbd2cSJim Jagielski             }
424*b1cdbd2cSJim Jagielski             break;
425*b1cdbd2cSJim Jagielski             case EXC_CACHEDVAL_ERROR:
426*b1cdbd2cSJim Jagielski             {
427*b1cdbd2cSJim Jagielski                 double fError = XclTools::ErrorToDouble( p->GetXclError() );
428*b1cdbd2cSJim Jagielski                 ScExternalRefCache::TokenRef pToken(new formula::FormulaDoubleToken(fError));
429*b1cdbd2cSJim Jagielski                 pCacheTable->setCell(rAddr.mnCol, rAddr.mnRow, pToken);
430*b1cdbd2cSJim Jagielski             }
431*b1cdbd2cSJim Jagielski             break;
432*b1cdbd2cSJim Jagielski             case EXC_CACHEDVAL_STRING:
433*b1cdbd2cSJim Jagielski             {
434*b1cdbd2cSJim Jagielski                 const String& rStr = p->GetString();
435*b1cdbd2cSJim Jagielski                 ScExternalRefCache::TokenRef pToken(new formula::FormulaStringToken(rStr));
436*b1cdbd2cSJim Jagielski                 pCacheTable->setCell(rAddr.mnCol, rAddr.mnRow, pToken);
437*b1cdbd2cSJim Jagielski             }
438*b1cdbd2cSJim Jagielski             break;
439*b1cdbd2cSJim Jagielski             default:
440*b1cdbd2cSJim Jagielski                 ;
441*b1cdbd2cSJim Jagielski         }
442*b1cdbd2cSJim Jagielski     }
443*b1cdbd2cSJim Jagielski }
444*b1cdbd2cSJim Jagielski 
445*b1cdbd2cSJim Jagielski // External document (SUPBOOK) ================================================
446*b1cdbd2cSJim Jagielski 
XclImpSupbook(XclImpStream & rStrm)447*b1cdbd2cSJim Jagielski XclImpSupbook::XclImpSupbook( XclImpStream& rStrm ) :
448*b1cdbd2cSJim Jagielski     XclImpRoot( rStrm.GetRoot() ),
449*b1cdbd2cSJim Jagielski     meType( EXC_SBTYPE_UNKNOWN ),
450*b1cdbd2cSJim Jagielski     mnSBTab( EXC_TAB_DELETED )
451*b1cdbd2cSJim Jagielski {
452*b1cdbd2cSJim Jagielski     sal_uInt16 nSBTabCnt;
453*b1cdbd2cSJim Jagielski     rStrm >> nSBTabCnt;
454*b1cdbd2cSJim Jagielski 
455*b1cdbd2cSJim Jagielski     if( rStrm.GetRecLeft() == 2 )
456*b1cdbd2cSJim Jagielski     {
457*b1cdbd2cSJim Jagielski         switch( rStrm.ReaduInt16() )
458*b1cdbd2cSJim Jagielski         {
459*b1cdbd2cSJim Jagielski             case EXC_SUPB_SELF:     meType = EXC_SBTYPE_SELF;   break;
460*b1cdbd2cSJim Jagielski             case EXC_SUPB_ADDIN:    meType = EXC_SBTYPE_ADDIN;  break;
461*b1cdbd2cSJim Jagielski             default:    DBG_ERRORFILE( "XclImpSupbook::XclImpSupbook - unknown special SUPBOOK type" );
462*b1cdbd2cSJim Jagielski         }
463*b1cdbd2cSJim Jagielski         return;
464*b1cdbd2cSJim Jagielski     }
465*b1cdbd2cSJim Jagielski 
466*b1cdbd2cSJim Jagielski     String aEncUrl( rStrm.ReadUniString() );
467*b1cdbd2cSJim Jagielski     bool bSelf = false;
468*b1cdbd2cSJim Jagielski     XclImpUrlHelper::DecodeUrl( maXclUrl, bSelf, GetRoot(), aEncUrl );
469*b1cdbd2cSJim Jagielski 
470*b1cdbd2cSJim Jagielski     if( maXclUrl.EqualsIgnoreCaseAscii( "\010EUROTOOL.XLA" ) )
471*b1cdbd2cSJim Jagielski     {
472*b1cdbd2cSJim Jagielski         meType = EXC_SBTYPE_EUROTOOL;
473*b1cdbd2cSJim Jagielski         maSupbTabList.Append( new XclImpSupbookTab( maXclUrl ) );
474*b1cdbd2cSJim Jagielski     }
475*b1cdbd2cSJim Jagielski     else if( nSBTabCnt )
476*b1cdbd2cSJim Jagielski     {
477*b1cdbd2cSJim Jagielski         meType = EXC_SBTYPE_EXTERN;
478*b1cdbd2cSJim Jagielski         for( sal_uInt16 nSBTab = 0; nSBTab < nSBTabCnt; ++nSBTab )
479*b1cdbd2cSJim Jagielski         {
480*b1cdbd2cSJim Jagielski             String aTabName( rStrm.ReadUniString() );
481*b1cdbd2cSJim Jagielski             maSupbTabList.Append( new XclImpSupbookTab( aTabName ) );
482*b1cdbd2cSJim Jagielski         }
483*b1cdbd2cSJim Jagielski     }
484*b1cdbd2cSJim Jagielski     else
485*b1cdbd2cSJim Jagielski     {
486*b1cdbd2cSJim Jagielski         meType = EXC_SBTYPE_SPECIAL;
487*b1cdbd2cSJim Jagielski         // create dummy list entry
488*b1cdbd2cSJim Jagielski         maSupbTabList.Append( new XclImpSupbookTab( maXclUrl ) );
489*b1cdbd2cSJim Jagielski     }
490*b1cdbd2cSJim Jagielski }
491*b1cdbd2cSJim Jagielski 
ReadXct(XclImpStream & rStrm)492*b1cdbd2cSJim Jagielski void XclImpSupbook::ReadXct( XclImpStream& rStrm )
493*b1cdbd2cSJim Jagielski {
494*b1cdbd2cSJim Jagielski     rStrm.Ignore( 2 );
495*b1cdbd2cSJim Jagielski     rStrm >> mnSBTab;
496*b1cdbd2cSJim Jagielski }
497*b1cdbd2cSJim Jagielski 
ReadCrn(XclImpStream & rStrm)498*b1cdbd2cSJim Jagielski void XclImpSupbook::ReadCrn( XclImpStream& rStrm )
499*b1cdbd2cSJim Jagielski {
500*b1cdbd2cSJim Jagielski     if( XclImpSupbookTab* pSBTab = maSupbTabList.GetObject( mnSBTab ) )
501*b1cdbd2cSJim Jagielski     {
502*b1cdbd2cSJim Jagielski         sal_uInt8 nXclColLast, nXclColFirst;
503*b1cdbd2cSJim Jagielski         sal_uInt16 nXclRow;
504*b1cdbd2cSJim Jagielski         rStrm >> nXclColLast >> nXclColFirst >> nXclRow;
505*b1cdbd2cSJim Jagielski 
506*b1cdbd2cSJim Jagielski         for( sal_uInt8 nXclCol = nXclColFirst; (nXclCol <= nXclColLast) && (rStrm.GetRecLeft() > 1); ++nXclCol )
507*b1cdbd2cSJim Jagielski             pSBTab->ReadCrn( rStrm, XclAddress( nXclCol, nXclRow ) );
508*b1cdbd2cSJim Jagielski     }
509*b1cdbd2cSJim Jagielski }
510*b1cdbd2cSJim Jagielski 
ReadExternname(XclImpStream & rStrm,ExcelToSc * pFormulaConv)511*b1cdbd2cSJim Jagielski void XclImpSupbook::ReadExternname( XclImpStream& rStrm, ExcelToSc* pFormulaConv )
512*b1cdbd2cSJim Jagielski {
513*b1cdbd2cSJim Jagielski     maExtNameList.Append( new XclImpExtName( *this, rStrm, meType, pFormulaConv ) );
514*b1cdbd2cSJim Jagielski }
515*b1cdbd2cSJim Jagielski 
GetExternName(sal_uInt16 nXclIndex) const516*b1cdbd2cSJim Jagielski const XclImpExtName* XclImpSupbook::GetExternName( sal_uInt16 nXclIndex ) const
517*b1cdbd2cSJim Jagielski {
518*b1cdbd2cSJim Jagielski     DBG_ASSERT( nXclIndex > 0, "XclImpSupbook::GetExternName - index must be >0" );
519*b1cdbd2cSJim Jagielski     return (meType == EXC_SBTYPE_SELF) ? 0 : maExtNameList.GetObject( nXclIndex - 1 );
520*b1cdbd2cSJim Jagielski }
521*b1cdbd2cSJim Jagielski 
GetLinkData(String & rApplic,String & rTopic) const522*b1cdbd2cSJim Jagielski bool XclImpSupbook::GetLinkData( String& rApplic, String& rTopic ) const
523*b1cdbd2cSJim Jagielski {
524*b1cdbd2cSJim Jagielski     return (meType == EXC_SBTYPE_SPECIAL) && XclImpUrlHelper::DecodeLink( rApplic, rTopic, maXclUrl );
525*b1cdbd2cSJim Jagielski }
526*b1cdbd2cSJim Jagielski 
GetMacroName(sal_uInt16 nXclNameIdx) const527*b1cdbd2cSJim Jagielski const String& XclImpSupbook::GetMacroName( sal_uInt16 nXclNameIdx ) const
528*b1cdbd2cSJim Jagielski {
529*b1cdbd2cSJim Jagielski     DBG_ASSERT( nXclNameIdx > 0, "XclImpSupbook::GetMacroName - index must be >0" );
530*b1cdbd2cSJim Jagielski     const XclImpName* pName = (meType == EXC_SBTYPE_SELF) ? GetNameManager().GetName( nXclNameIdx ) : 0;
531*b1cdbd2cSJim Jagielski     return (pName && pName->IsVBName()) ? pName->GetScName() : EMPTY_STRING;
532*b1cdbd2cSJim Jagielski }
533*b1cdbd2cSJim Jagielski 
GetTabName(sal_uInt16 nXtiTab) const534*b1cdbd2cSJim Jagielski const String& XclImpSupbook::GetTabName( sal_uInt16 nXtiTab ) const
535*b1cdbd2cSJim Jagielski {
536*b1cdbd2cSJim Jagielski     if (maSupbTabList.Empty())
537*b1cdbd2cSJim Jagielski         return EMPTY_STRING;
538*b1cdbd2cSJim Jagielski 
539*b1cdbd2cSJim Jagielski     sal_uInt16 i = 0;
540*b1cdbd2cSJim Jagielski     for (XclImpSupbookTab* p = maSupbTabList.First(); p; p = maSupbTabList.Next(), ++i)
541*b1cdbd2cSJim Jagielski     {
542*b1cdbd2cSJim Jagielski         if (i == nXtiTab)
543*b1cdbd2cSJim Jagielski             return p->GetTabName();
544*b1cdbd2cSJim Jagielski     }
545*b1cdbd2cSJim Jagielski 
546*b1cdbd2cSJim Jagielski     return EMPTY_STRING;
547*b1cdbd2cSJim Jagielski }
548*b1cdbd2cSJim Jagielski 
GetTabCount() const549*b1cdbd2cSJim Jagielski sal_uInt16 XclImpSupbook::GetTabCount() const
550*b1cdbd2cSJim Jagielski {
551*b1cdbd2cSJim Jagielski     return ulimit_cast<sal_uInt16>(maSupbTabList.Count());
552*b1cdbd2cSJim Jagielski }
553*b1cdbd2cSJim Jagielski 
LoadCachedValues()554*b1cdbd2cSJim Jagielski void XclImpSupbook::LoadCachedValues()
555*b1cdbd2cSJim Jagielski {
556*b1cdbd2cSJim Jagielski     if (meType != EXC_SBTYPE_EXTERN || GetExtDocOptions().GetDocSettings().mnLinkCnt > 0 || !GetDocShell())
557*b1cdbd2cSJim Jagielski         return;
558*b1cdbd2cSJim Jagielski 
559*b1cdbd2cSJim Jagielski     String aAbsUrl( ScGlobal::GetAbsDocName(maXclUrl, GetDocShell()) );
560*b1cdbd2cSJim Jagielski 
561*b1cdbd2cSJim Jagielski     ScExternalRefManager* pRefMgr = GetRoot().GetDoc().GetExternalRefManager();
562*b1cdbd2cSJim Jagielski     sal_uInt16 nFileId = pRefMgr->getExternalFileId(aAbsUrl);
563*b1cdbd2cSJim Jagielski 
564*b1cdbd2cSJim Jagielski     sal_uInt16 nCount = static_cast< sal_uInt16 >( maSupbTabList.Count() );
565*b1cdbd2cSJim Jagielski     for (sal_uInt16 i = 0; i < nCount; ++i)
566*b1cdbd2cSJim Jagielski     {
567*b1cdbd2cSJim Jagielski         XclImpSupbookTab* pTab = maSupbTabList.GetObject(i);
568*b1cdbd2cSJim Jagielski         if (!pTab)
569*b1cdbd2cSJim Jagielski             return;
570*b1cdbd2cSJim Jagielski 
571*b1cdbd2cSJim Jagielski         const String& rTabName = pTab->GetTabName();
572*b1cdbd2cSJim Jagielski         ScExternalRefCache::TableTypeRef pCacheTable = pRefMgr->getCacheTable(nFileId, rTabName, true);
573*b1cdbd2cSJim Jagielski         pTab->LoadCachedValues(pCacheTable);
574*b1cdbd2cSJim Jagielski         pCacheTable->setWholeTableCached();
575*b1cdbd2cSJim Jagielski     }
576*b1cdbd2cSJim Jagielski }
577*b1cdbd2cSJim Jagielski 
578*b1cdbd2cSJim Jagielski // Import link manager ========================================================
579*b1cdbd2cSJim Jagielski 
XclImpLinkManagerImpl(const XclImpRoot & rRoot)580*b1cdbd2cSJim Jagielski XclImpLinkManagerImpl::XclImpLinkManagerImpl( const XclImpRoot& rRoot ) :
581*b1cdbd2cSJim Jagielski     XclImpRoot( rRoot ),
582*b1cdbd2cSJim Jagielski     mbCreated( false )
583*b1cdbd2cSJim Jagielski {
584*b1cdbd2cSJim Jagielski }
585*b1cdbd2cSJim Jagielski 
ReadExternsheet(XclImpStream & rStrm)586*b1cdbd2cSJim Jagielski void XclImpLinkManagerImpl::ReadExternsheet( XclImpStream& rStrm )
587*b1cdbd2cSJim Jagielski {
588*b1cdbd2cSJim Jagielski     sal_uInt16 nXtiCount;
589*b1cdbd2cSJim Jagielski     rStrm >> nXtiCount;
590*b1cdbd2cSJim Jagielski     DBG_ASSERT( static_cast< sal_Size >( nXtiCount * 6 ) == rStrm.GetRecLeft(), "XclImpLinkManagerImpl::ReadExternsheet - invalid count" );
591*b1cdbd2cSJim Jagielski     nXtiCount = static_cast< sal_uInt16 >( ::std::min< sal_Size >( nXtiCount, rStrm.GetRecLeft() / 6 ) );
592*b1cdbd2cSJim Jagielski 
593*b1cdbd2cSJim Jagielski     /*  #i104057# A weird external XLS generator writes multiple EXTERNSHEET
594*b1cdbd2cSJim Jagielski         records instead of only one as expected. Surprisingly, Excel seems to
595*b1cdbd2cSJim Jagielski         insert the entries of the second record before the entries of the first
596*b1cdbd2cSJim Jagielski         record. */
597*b1cdbd2cSJim Jagielski     XclImpXtiVector aNewEntries( nXtiCount );
598*b1cdbd2cSJim Jagielski     for( XclImpXtiVector::iterator aIt = aNewEntries.begin(), aEnd = aNewEntries.end(); rStrm.IsValid() && (aIt != aEnd); ++aIt )
599*b1cdbd2cSJim Jagielski         rStrm >> *aIt;
600*b1cdbd2cSJim Jagielski     maXtiList.insert( maXtiList.begin(), aNewEntries.begin(), aNewEntries.end() );
601*b1cdbd2cSJim Jagielski 
602*b1cdbd2cSJim Jagielski     LoadCachedValues();
603*b1cdbd2cSJim Jagielski }
604*b1cdbd2cSJim Jagielski 
ReadSupbook(XclImpStream & rStrm)605*b1cdbd2cSJim Jagielski void XclImpLinkManagerImpl::ReadSupbook( XclImpStream& rStrm )
606*b1cdbd2cSJim Jagielski {
607*b1cdbd2cSJim Jagielski     maSupbookList.Append( new XclImpSupbook( rStrm ) );
608*b1cdbd2cSJim Jagielski }
609*b1cdbd2cSJim Jagielski 
ReadXct(XclImpStream & rStrm)610*b1cdbd2cSJim Jagielski void XclImpLinkManagerImpl::ReadXct( XclImpStream& rStrm )
611*b1cdbd2cSJim Jagielski {
612*b1cdbd2cSJim Jagielski     if( XclImpSupbook* pSupbook = maSupbookList.Last() )
613*b1cdbd2cSJim Jagielski         pSupbook->ReadXct( rStrm );
614*b1cdbd2cSJim Jagielski }
615*b1cdbd2cSJim Jagielski 
ReadCrn(XclImpStream & rStrm)616*b1cdbd2cSJim Jagielski void XclImpLinkManagerImpl::ReadCrn( XclImpStream& rStrm )
617*b1cdbd2cSJim Jagielski {
618*b1cdbd2cSJim Jagielski     if( XclImpSupbook* pSupbook = maSupbookList.Last() )
619*b1cdbd2cSJim Jagielski         pSupbook->ReadCrn( rStrm );
620*b1cdbd2cSJim Jagielski }
621*b1cdbd2cSJim Jagielski 
ReadExternname(XclImpStream & rStrm,ExcelToSc * pFormulaConv)622*b1cdbd2cSJim Jagielski void XclImpLinkManagerImpl::ReadExternname( XclImpStream& rStrm, ExcelToSc* pFormulaConv )
623*b1cdbd2cSJim Jagielski {
624*b1cdbd2cSJim Jagielski     if( XclImpSupbook* pSupbook = maSupbookList.Last() )
625*b1cdbd2cSJim Jagielski         pSupbook->ReadExternname( rStrm, pFormulaConv );
626*b1cdbd2cSJim Jagielski }
627*b1cdbd2cSJim Jagielski 
IsSelfRef(sal_uInt16 nXtiIndex) const628*b1cdbd2cSJim Jagielski bool XclImpLinkManagerImpl::IsSelfRef( sal_uInt16 nXtiIndex ) const
629*b1cdbd2cSJim Jagielski {
630*b1cdbd2cSJim Jagielski     const XclImpSupbook* pSupbook = GetSupbook( nXtiIndex );
631*b1cdbd2cSJim Jagielski     return pSupbook && (pSupbook->GetType() == EXC_SBTYPE_SELF);
632*b1cdbd2cSJim Jagielski }
633*b1cdbd2cSJim Jagielski 
GetScTabRange(SCTAB & rnFirstScTab,SCTAB & rnLastScTab,sal_uInt16 nXtiIndex) const634*b1cdbd2cSJim Jagielski bool XclImpLinkManagerImpl::GetScTabRange(
635*b1cdbd2cSJim Jagielski         SCTAB& rnFirstScTab, SCTAB& rnLastScTab, sal_uInt16 nXtiIndex ) const
636*b1cdbd2cSJim Jagielski {
637*b1cdbd2cSJim Jagielski     if( const XclImpXti* pXti = GetXti( nXtiIndex ) )
638*b1cdbd2cSJim Jagielski     {
639*b1cdbd2cSJim Jagielski         if (maSupbookList.GetObject(pXti->mnSupbook))
640*b1cdbd2cSJim Jagielski         {
641*b1cdbd2cSJim Jagielski             rnFirstScTab = pXti->mnSBTabFirst;
642*b1cdbd2cSJim Jagielski             rnLastScTab  = pXti->mnSBTabLast;
643*b1cdbd2cSJim Jagielski             return true;
644*b1cdbd2cSJim Jagielski         }
645*b1cdbd2cSJim Jagielski     }
646*b1cdbd2cSJim Jagielski     return false;
647*b1cdbd2cSJim Jagielski }
648*b1cdbd2cSJim Jagielski 
GetExternName(sal_uInt16 nXtiIndex,sal_uInt16 nExtName) const649*b1cdbd2cSJim Jagielski const XclImpExtName* XclImpLinkManagerImpl::GetExternName( sal_uInt16 nXtiIndex, sal_uInt16 nExtName ) const
650*b1cdbd2cSJim Jagielski {
651*b1cdbd2cSJim Jagielski     const XclImpSupbook* pSupbook = GetSupbook( nXtiIndex );
652*b1cdbd2cSJim Jagielski     return pSupbook ? pSupbook->GetExternName( nExtName ) : 0;
653*b1cdbd2cSJim Jagielski }
654*b1cdbd2cSJim Jagielski 
GetSupbookUrl(sal_uInt16 nXtiIndex) const655*b1cdbd2cSJim Jagielski const String* XclImpLinkManagerImpl::GetSupbookUrl( sal_uInt16 nXtiIndex ) const
656*b1cdbd2cSJim Jagielski {
657*b1cdbd2cSJim Jagielski     const XclImpSupbook* p = GetSupbook( nXtiIndex );
658*b1cdbd2cSJim Jagielski     if (!p)
659*b1cdbd2cSJim Jagielski         return NULL;
660*b1cdbd2cSJim Jagielski     return &p->GetXclUrl();
661*b1cdbd2cSJim Jagielski }
662*b1cdbd2cSJim Jagielski 
GetSupbookTabName(sal_uInt16 nXti,sal_uInt16 nXtiTab) const663*b1cdbd2cSJim Jagielski const String& XclImpLinkManagerImpl::GetSupbookTabName( sal_uInt16 nXti, sal_uInt16 nXtiTab ) const
664*b1cdbd2cSJim Jagielski {
665*b1cdbd2cSJim Jagielski     const XclImpSupbook* p = GetSupbook(nXti);
666*b1cdbd2cSJim Jagielski     return p ? p->GetTabName(nXtiTab) : EMPTY_STRING;
667*b1cdbd2cSJim Jagielski }
668*b1cdbd2cSJim Jagielski 
GetLinkData(String & rApplic,String & rTopic,sal_uInt16 nXtiIndex) const669*b1cdbd2cSJim Jagielski bool XclImpLinkManagerImpl::GetLinkData( String& rApplic, String& rTopic, sal_uInt16 nXtiIndex ) const
670*b1cdbd2cSJim Jagielski {
671*b1cdbd2cSJim Jagielski     const XclImpSupbook* pSupbook = GetSupbook( nXtiIndex );
672*b1cdbd2cSJim Jagielski     return pSupbook && pSupbook->GetLinkData( rApplic, rTopic );
673*b1cdbd2cSJim Jagielski }
674*b1cdbd2cSJim Jagielski 
GetMacroName(sal_uInt16 nExtSheet,sal_uInt16 nExtName) const675*b1cdbd2cSJim Jagielski const String& XclImpLinkManagerImpl::GetMacroName( sal_uInt16 nExtSheet, sal_uInt16 nExtName ) const
676*b1cdbd2cSJim Jagielski {
677*b1cdbd2cSJim Jagielski     const XclImpSupbook* pSupbook = GetSupbook( nExtSheet );
678*b1cdbd2cSJim Jagielski     return pSupbook ? pSupbook->GetMacroName( nExtName ) : EMPTY_STRING;
679*b1cdbd2cSJim Jagielski }
680*b1cdbd2cSJim Jagielski 
GetXti(sal_uInt16 nXtiIndex) const681*b1cdbd2cSJim Jagielski const XclImpXti* XclImpLinkManagerImpl::GetXti( sal_uInt16 nXtiIndex ) const
682*b1cdbd2cSJim Jagielski {
683*b1cdbd2cSJim Jagielski     return (nXtiIndex < maXtiList.size()) ? &maXtiList[ nXtiIndex ] : 0;
684*b1cdbd2cSJim Jagielski }
685*b1cdbd2cSJim Jagielski 
GetSupbook(sal_uInt16 nXtiIndex) const686*b1cdbd2cSJim Jagielski const XclImpSupbook* XclImpLinkManagerImpl::GetSupbook( sal_uInt16 nXtiIndex ) const
687*b1cdbd2cSJim Jagielski {
688*b1cdbd2cSJim Jagielski     const XclImpXti* pXti = GetXti( nXtiIndex );
689*b1cdbd2cSJim Jagielski     return pXti ? maSupbookList.GetObject( pXti->mnSupbook ) : 0;
690*b1cdbd2cSJim Jagielski }
691*b1cdbd2cSJim Jagielski 
692*b1cdbd2cSJim Jagielski //UNUSED2009-05 const XclImpSupbook* XclImpLinkManagerImpl::GetSupbook( const String& rUrl ) const
693*b1cdbd2cSJim Jagielski //UNUSED2009-05 {
694*b1cdbd2cSJim Jagielski //UNUSED2009-05     for( const XclImpSupbook* pSupbook = maSupbookList.First(); pSupbook; pSupbook = maSupbookList.Next() )
695*b1cdbd2cSJim Jagielski //UNUSED2009-05         if( pSupbook->GetXclUrl() == rUrl )
696*b1cdbd2cSJim Jagielski //UNUSED2009-05             return pSupbook;
697*b1cdbd2cSJim Jagielski //UNUSED2009-05     return 0;
698*b1cdbd2cSJim Jagielski //UNUSED2009-05 }
699*b1cdbd2cSJim Jagielski 
LoadCachedValues()700*b1cdbd2cSJim Jagielski void XclImpLinkManagerImpl::LoadCachedValues()
701*b1cdbd2cSJim Jagielski {
702*b1cdbd2cSJim Jagielski     // Read all CRN records which can be accessed via XclImpSupbook, and store
703*b1cdbd2cSJim Jagielski     // the cached values to the external reference manager.
704*b1cdbd2cSJim Jagielski 
705*b1cdbd2cSJim Jagielski     sal_uInt32 nCount = maSupbookList.Count();
706*b1cdbd2cSJim Jagielski     for (sal_uInt16 nSupbook = 0; nSupbook < nCount; ++nSupbook)
707*b1cdbd2cSJim Jagielski     {
708*b1cdbd2cSJim Jagielski         XclImpSupbook* pSupbook = maSupbookList.GetObject(nSupbook);
709*b1cdbd2cSJim Jagielski         pSupbook->LoadCachedValues();
710*b1cdbd2cSJim Jagielski     }
711*b1cdbd2cSJim Jagielski }
712*b1cdbd2cSJim Jagielski 
713*b1cdbd2cSJim Jagielski //UNUSED2009-05 bool XclImpLinkManagerImpl::FindNextTabRange(
714*b1cdbd2cSJim Jagielski //UNUSED2009-05         sal_uInt16& rnSBTabFirst, sal_uInt16& rnSBTabLast,
715*b1cdbd2cSJim Jagielski //UNUSED2009-05         sal_uInt16 nSupbook, sal_uInt16 nSBTabStart ) const
716*b1cdbd2cSJim Jagielski //UNUSED2009-05 {
717*b1cdbd2cSJim Jagielski //UNUSED2009-05     rnSBTabFirst = rnSBTabLast = EXC_NOTAB;
718*b1cdbd2cSJim Jagielski //UNUSED2009-05     for( const XclImpXti* pXti = maXtiList.First(); pXti; pXti = maXtiList.Next() )
719*b1cdbd2cSJim Jagielski //UNUSED2009-05     {
720*b1cdbd2cSJim Jagielski //UNUSED2009-05         if( (nSupbook == pXti->mnSupbook) && (nSBTabStart <= pXti->mnSBTabLast) && (pXti->mnSBTabFirst < rnSBTabFirst) )
721*b1cdbd2cSJim Jagielski //UNUSED2009-05         {
722*b1cdbd2cSJim Jagielski //UNUSED2009-05             rnSBTabFirst = ::std::max( nSBTabStart, pXti->mnSBTabFirst );
723*b1cdbd2cSJim Jagielski //UNUSED2009-05             rnSBTabLast = pXti->mnSBTabLast;
724*b1cdbd2cSJim Jagielski //UNUSED2009-05         }
725*b1cdbd2cSJim Jagielski //UNUSED2009-05     }
726*b1cdbd2cSJim Jagielski //UNUSED2009-05     return rnSBTabFirst != EXC_NOTAB;
727*b1cdbd2cSJim Jagielski //UNUSED2009-05 }
728*b1cdbd2cSJim Jagielski 
729*b1cdbd2cSJim Jagielski // ============================================================================
730*b1cdbd2cSJim Jagielski 
XclImpLinkManager(const XclImpRoot & rRoot)731*b1cdbd2cSJim Jagielski XclImpLinkManager::XclImpLinkManager( const XclImpRoot& rRoot ) :
732*b1cdbd2cSJim Jagielski     XclImpRoot( rRoot ),
733*b1cdbd2cSJim Jagielski     mxImpl( new XclImpLinkManagerImpl( rRoot ) )
734*b1cdbd2cSJim Jagielski {
735*b1cdbd2cSJim Jagielski }
736*b1cdbd2cSJim Jagielski 
~XclImpLinkManager()737*b1cdbd2cSJim Jagielski XclImpLinkManager::~XclImpLinkManager()
738*b1cdbd2cSJim Jagielski {
739*b1cdbd2cSJim Jagielski }
740*b1cdbd2cSJim Jagielski 
ReadExternsheet(XclImpStream & rStrm)741*b1cdbd2cSJim Jagielski void XclImpLinkManager::ReadExternsheet( XclImpStream& rStrm )
742*b1cdbd2cSJim Jagielski {
743*b1cdbd2cSJim Jagielski     mxImpl->ReadExternsheet( rStrm );
744*b1cdbd2cSJim Jagielski }
745*b1cdbd2cSJim Jagielski 
ReadSupbook(XclImpStream & rStrm)746*b1cdbd2cSJim Jagielski void XclImpLinkManager::ReadSupbook( XclImpStream& rStrm )
747*b1cdbd2cSJim Jagielski {
748*b1cdbd2cSJim Jagielski     mxImpl->ReadSupbook( rStrm );
749*b1cdbd2cSJim Jagielski }
750*b1cdbd2cSJim Jagielski 
ReadXct(XclImpStream & rStrm)751*b1cdbd2cSJim Jagielski void XclImpLinkManager::ReadXct( XclImpStream& rStrm )
752*b1cdbd2cSJim Jagielski {
753*b1cdbd2cSJim Jagielski     mxImpl->ReadXct( rStrm );
754*b1cdbd2cSJim Jagielski }
755*b1cdbd2cSJim Jagielski 
ReadCrn(XclImpStream & rStrm)756*b1cdbd2cSJim Jagielski void XclImpLinkManager::ReadCrn( XclImpStream& rStrm )
757*b1cdbd2cSJim Jagielski {
758*b1cdbd2cSJim Jagielski     mxImpl->ReadCrn( rStrm );
759*b1cdbd2cSJim Jagielski }
760*b1cdbd2cSJim Jagielski 
ReadExternname(XclImpStream & rStrm,ExcelToSc * pFormulaConv)761*b1cdbd2cSJim Jagielski void XclImpLinkManager::ReadExternname( XclImpStream& rStrm, ExcelToSc* pFormulaConv )
762*b1cdbd2cSJim Jagielski {
763*b1cdbd2cSJim Jagielski     mxImpl->ReadExternname( rStrm, pFormulaConv );
764*b1cdbd2cSJim Jagielski }
765*b1cdbd2cSJim Jagielski 
IsSelfRef(sal_uInt16 nXtiIndex) const766*b1cdbd2cSJim Jagielski bool XclImpLinkManager::IsSelfRef( sal_uInt16 nXtiIndex ) const
767*b1cdbd2cSJim Jagielski {
768*b1cdbd2cSJim Jagielski     return mxImpl->IsSelfRef( nXtiIndex );
769*b1cdbd2cSJim Jagielski }
770*b1cdbd2cSJim Jagielski 
GetScTabRange(SCTAB & rnFirstScTab,SCTAB & rnLastScTab,sal_uInt16 nXtiIndex) const771*b1cdbd2cSJim Jagielski bool XclImpLinkManager::GetScTabRange(
772*b1cdbd2cSJim Jagielski         SCTAB& rnFirstScTab, SCTAB& rnLastScTab, sal_uInt16 nXtiIndex ) const
773*b1cdbd2cSJim Jagielski {
774*b1cdbd2cSJim Jagielski     return mxImpl->GetScTabRange( rnFirstScTab, rnLastScTab, nXtiIndex );
775*b1cdbd2cSJim Jagielski }
776*b1cdbd2cSJim Jagielski 
GetExternName(sal_uInt16 nXtiIndex,sal_uInt16 nExtName) const777*b1cdbd2cSJim Jagielski const XclImpExtName* XclImpLinkManager::GetExternName( sal_uInt16 nXtiIndex, sal_uInt16 nExtName ) const
778*b1cdbd2cSJim Jagielski {
779*b1cdbd2cSJim Jagielski     return mxImpl->GetExternName( nXtiIndex, nExtName );
780*b1cdbd2cSJim Jagielski }
781*b1cdbd2cSJim Jagielski 
GetSupbookUrl(sal_uInt16 nXtiIndex) const782*b1cdbd2cSJim Jagielski const String* XclImpLinkManager::GetSupbookUrl( sal_uInt16 nXtiIndex ) const
783*b1cdbd2cSJim Jagielski {
784*b1cdbd2cSJim Jagielski     return mxImpl->GetSupbookUrl(nXtiIndex);
785*b1cdbd2cSJim Jagielski }
786*b1cdbd2cSJim Jagielski 
GetSupbookTabName(sal_uInt16 nXti,sal_uInt16 nXtiTab) const787*b1cdbd2cSJim Jagielski const String& XclImpLinkManager::GetSupbookTabName( sal_uInt16 nXti,  sal_uInt16 nXtiTab ) const
788*b1cdbd2cSJim Jagielski {
789*b1cdbd2cSJim Jagielski     return mxImpl->GetSupbookTabName(nXti, nXtiTab);
790*b1cdbd2cSJim Jagielski }
791*b1cdbd2cSJim Jagielski 
GetLinkData(String & rApplic,String & rTopic,sal_uInt16 nXtiIndex) const792*b1cdbd2cSJim Jagielski bool XclImpLinkManager::GetLinkData( String& rApplic, String& rTopic, sal_uInt16 nXtiIndex ) const
793*b1cdbd2cSJim Jagielski {
794*b1cdbd2cSJim Jagielski     return mxImpl->GetLinkData( rApplic, rTopic, nXtiIndex );
795*b1cdbd2cSJim Jagielski }
796*b1cdbd2cSJim Jagielski 
GetMacroName(sal_uInt16 nExtSheet,sal_uInt16 nExtName) const797*b1cdbd2cSJim Jagielski const String& XclImpLinkManager::GetMacroName( sal_uInt16 nExtSheet, sal_uInt16 nExtName ) const
798*b1cdbd2cSJim Jagielski {
799*b1cdbd2cSJim Jagielski     return mxImpl->GetMacroName( nExtSheet, nExtName );
800*b1cdbd2cSJim Jagielski }
801*b1cdbd2cSJim Jagielski 
802*b1cdbd2cSJim Jagielski // ============================================================================
803*b1cdbd2cSJim Jagielski 
804