1 /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
3 * This file is part of the LibreOffice project.
5 * This Source Code Form is subject to the terms of the Mozilla Public
6 * License, v. 2.0. If a copy of the MPL was not distributed with this
7 * file, You can obtain one at http://mozilla.org/MPL/2.0/.
9 * This file incorporates work covered by the following license notice:
11 * Licensed to the Apache Software Foundation (ASF) under one or more
12 * contributor license agreements. See the NOTICE file distributed
13 * with this work for additional information regarding copyright
14 * ownership. The ASF licenses this file to you under the Apache
15 * License, Version 2.0 (the "License"); you may not use this file
16 * except in compliance with the License. You may obtain a copy of
17 * the License at http://www.apache.org/licenses/LICENSE-2.0 .
20 #include "xmlsubti.hxx"
22 #include "xmlstyli.hxx"
23 #include "xmlimprt.hxx"
24 #include "document.hxx"
25 #include "markdata.hxx"
26 #include "XMLConverter.hxx"
28 #include "cellsuno.hxx"
29 #include "XMLStylesImportHelper.hxx"
30 #include "sheetdata.hxx"
31 #include "tabprotection.hxx"
32 #include "tokenarray.hxx"
33 #include "convuno.hxx"
34 #include "documentimport.hxx"
36 #include <svx/svdpage.hxx>
38 #include <sax/tools/converter.hxx>
39 #include <xmloff/xmltkmap.hxx>
40 #include <xmloff/nmspmap.hxx>
41 #include <xmloff/xmlerror.hxx>
42 #include <com/sun/star/sheet/XSpreadsheetDocument.hpp>
43 #include <com/sun/star/sheet/XSheetCellRange.hpp>
44 #include <com/sun/star/sheet/XCellRangeAddressable.hpp>
45 #include <com/sun/star/sheet/CellInsertMode.hpp>
46 #include <com/sun/star/sheet/XCellRangeMovement.hpp>
47 #include <com/sun/star/drawing/XDrawPageSupplier.hpp>
48 #include <com/sun/star/container/XNamed.hpp>
49 #include <com/sun/star/util/XProtectable.hpp>
50 #include <com/sun/star/sheet/XArrayFormulaRange.hpp>
52 #include <boost/scoped_ptr.hpp>
54 using namespace com::sun::star
;
56 ScXMLTabProtectionData::ScXMLTabProtectionData() :
57 meHash1(PASSHASH_SHA1
),
58 meHash2(PASSHASH_UNSPECIFIED
),
60 mbSelectProtectedCells(true),
61 mbSelectUnprotectedCells(true)
65 ScMyTables::ScMyTables(ScXMLImport
& rTempImport
)
66 : rImport(rTempImport
),
67 aFixupOLEs(rTempImport
),
68 maCurrentCellPos(ScAddress::INITIALIZE_INVALID
),
70 nCurrentDrawPage( -1 ),
75 ScMyTables::~ScMyTables()
81 uno::Reference
<sheet::XSpreadsheet
> getCurrentSheet(const uno::Reference
<frame::XModel
>& xModel
, SCTAB nSheet
)
83 uno::Reference
<sheet::XSpreadsheet
> xSheet
;
84 uno::Reference
<sheet::XSpreadsheetDocument
> xSpreadDoc(xModel
, uno::UNO_QUERY
);
88 uno::Reference
<sheet::XSpreadsheets
> xSheets(xSpreadDoc
->getSheets());
92 uno::Reference
<container::XIndexAccess
> xIndex(xSheets
, uno::UNO_QUERY
);
96 xSheet
.set(xIndex
->getByIndex(nSheet
), uno::UNO_QUERY
);
102 void ScMyTables::NewSheet(const OUString
& sTableName
, const OUString
& sStyleName
,
103 const ScXMLTabProtectionData
& rProtectData
)
105 if (rImport
.GetModel().is())
107 nCurrentColCount
= 0;
108 sCurrentSheetName
= sTableName
;
109 //reset cols and rows for new sheet, but increment tab
110 maCurrentCellPos
.SetCol(-1);
111 maCurrentCellPos
.SetRow(-1);
112 maCurrentCellPos
.SetTab(maCurrentCellPos
.Tab() + 1);
114 maProtectionData
= rProtectData
;
115 ScDocument
*pDoc
= ScXMLConverter::GetScDocument(rImport
.GetModel());
117 // The document contains one sheet when created. So for the first
118 // sheet, we only need to set its name.
119 if (maCurrentCellPos
.Tab() > 0)
120 pDoc
->AppendTabOnLoad(sTableName
);
122 pDoc
->SetTabNameOnLoad(maCurrentCellPos
.Tab(), sTableName
);
124 rImport
.SetTableStyle(sStyleName
);
125 xCurrentSheet
= getCurrentSheet(rImport
.GetModel(), maCurrentCellPos
.Tab());
126 if (xCurrentSheet
.is())
128 // We need to set the current cell range here regardless of
129 // presence of style name.
130 xCurrentCellRange
.set(xCurrentSheet
, uno::UNO_QUERY
);
131 SetTableStyle(sStyleName
);
136 void ScMyTables::SetTableStyle(const OUString
& sStyleName
)
138 //these uno calls are a bit difficult to remove, XMLTableStyleContext::FillPropertySet uses
139 //SvXMLImportPropertyMapper::FillPropertySet
140 if ( !sStyleName
.isEmpty() )
142 // #i57869# All table style properties for all sheets are now applied here,
143 // before importing the contents.
144 // This is needed for the background color.
145 // Sheet visibility has special handling in ScDocFunc::SetTableVisible to
146 // allow hiding the first sheet.
147 // RTL layout is only remembered, not actually applied, so the shapes can
148 // be loaded before mirroring.
150 if ( xCurrentSheet
.is() )
152 xCurrentCellRange
.set(xCurrentSheet
, uno::UNO_QUERY
);
153 uno::Reference
<beans::XPropertySet
> xProperties(xCurrentSheet
, uno::UNO_QUERY
);
154 if ( xProperties
.is() )
156 XMLTableStylesContext
*pStyles
= static_cast<XMLTableStylesContext
*>(rImport
.GetAutoStyles());
159 XMLTableStyleContext
* pStyle
= const_cast<XMLTableStyleContext
*>(static_cast<const XMLTableStyleContext
*>(pStyles
->FindStyleChildContext(
160 XML_STYLE_FAMILY_TABLE_TABLE
, sStyleName
, true)));
163 pStyle
->FillPropertySet(xProperties
);
165 ScSheetSaveData
* pSheetData
= ScModelObj::getImplementation(rImport
.GetModel())->GetSheetSaveData();
166 pSheetData
->AddTableStyle( sStyleName
, ScAddress( 0, 0, maCurrentCellPos
.Tab() ) );
174 void ScMyTables::AddRow()
176 maCurrentCellPos
.SetRow(maCurrentCellPos
.Row() + 1);
177 maCurrentCellPos
.SetCol(-1); //reset columns for new row
180 void ScMyTables::SetRowStyle(const OUString
& rCellStyleName
)
182 rImport
.GetStylesImportHelper()->SetRowStyle(rCellStyleName
);
185 void ScMyTables::AddColumn(bool bIsCovered
)
187 maCurrentCellPos
.SetCol( maCurrentCellPos
.Col() + 1 );
188 //here only need to set column style if this is the first row and
189 //the cell is not covered.
190 if(maCurrentCellPos
.Row() == 0 && !bIsCovered
)
191 rImport
.GetStylesImportHelper()->InsertCol(maCurrentCellPos
.Col(), maCurrentCellPos
.Tab(), rImport
.GetDocument());
194 void ScMyTables::DeleteTable()
196 ScXMLImport::MutexGuard
aGuard(rImport
);
198 rImport
.GetStylesImportHelper()->SetStylesToRanges();
199 rImport
.SetStylesToRangesFinished();
201 maMatrixRangeList
.RemoveAll();
203 if (rImport
.GetDocument() && maProtectionData
.mbProtected
)
205 uno::Sequence
<sal_Int8
> aHash
;
206 ::sax::Converter::decodeBase64(aHash
, maProtectionData
.maPassword
);
208 boost::scoped_ptr
<ScTableProtection
> pProtect(new ScTableProtection
);
209 pProtect
->setProtected(maProtectionData
.mbProtected
);
210 pProtect
->setPasswordHash(aHash
, maProtectionData
.meHash1
, maProtectionData
.meHash2
);
211 pProtect
->setOption(ScTableProtection::SELECT_LOCKED_CELLS
, maProtectionData
.mbSelectProtectedCells
);
212 pProtect
->setOption(ScTableProtection::SELECT_UNLOCKED_CELLS
, maProtectionData
.mbSelectUnprotectedCells
);
213 rImport
.GetDocument()->SetTabProtection(maCurrentCellPos
.Tab(), pProtect
.get());
217 void ScMyTables::AddColStyle(const sal_Int32 nRepeat
, const OUString
& rCellStyleName
)
219 rImport
.GetStylesImportHelper()->AddColumnStyle(rCellStyleName
, nCurrentColCount
, nRepeat
);
220 nCurrentColCount
+= nRepeat
;
221 SAL_WARN_IF(nCurrentColCount
> MAXCOLCOUNT
, "sc", "more columns than fit into SCCOL");
222 nCurrentColCount
= std::min
<sal_Int32
>( nCurrentColCount
, MAXCOLCOUNT
);
225 uno::Reference
< drawing::XDrawPage
> ScMyTables::GetCurrentXDrawPage()
227 if( (maCurrentCellPos
.Tab() != nCurrentDrawPage
) || !xDrawPage
.is() )
229 uno::Reference
<drawing::XDrawPageSupplier
> xDrawPageSupplier( xCurrentSheet
, uno::UNO_QUERY
);
230 if( xDrawPageSupplier
.is() )
231 xDrawPage
.set(xDrawPageSupplier
->getDrawPage());
232 nCurrentDrawPage
= sal::static_int_cast
<sal_Int16
>(maCurrentCellPos
.Tab());
237 uno::Reference
< drawing::XShapes
> ScMyTables::GetCurrentXShapes()
239 if( (maCurrentCellPos
.Tab() != nCurrentXShapes
) || !xShapes
.is() )
241 xShapes
.set(GetCurrentXDrawPage(), uno::UNO_QUERY
);
242 rImport
.GetShapeImport()->startPage(xShapes
);
243 rImport
.GetShapeImport()->pushGroupForSorting ( xShapes
);
244 nCurrentXShapes
= sal::static_int_cast
<sal_Int16
>(maCurrentCellPos
.Tab());
251 bool ScMyTables::HasDrawPage()
253 return !((maCurrentCellPos
.Tab() != nCurrentDrawPage
) || !xDrawPage
.is());
256 bool ScMyTables::HasXShapes()
258 return !((maCurrentCellPos
.Tab() != nCurrentXShapes
) || !xShapes
.is());
261 void ScMyTables::AddOLE(uno::Reference
<drawing::XShape
>& rShape
,
262 const OUString
&rRangeList
)
264 aFixupOLEs
.AddOLE(rShape
, rRangeList
);
267 void ScMyTables::AddMatrixRange(
268 const SCCOL nStartColumn
, const SCROW nStartRow
, const SCCOL nEndColumn
, const SCROW nEndRow
,
269 const OUString
& rFormula
, const OUString
& rFormulaNmsp
, const formula::FormulaGrammar::Grammar eGrammar
)
271 OSL_ENSURE(nEndRow
>= nStartRow
, "wrong row order");
272 OSL_ENSURE(nEndColumn
>= nStartColumn
, "wrong column order");
274 nStartColumn
, nStartRow
, maCurrentCellPos
.Tab(),
275 nEndColumn
, nEndRow
, maCurrentCellPos
.Tab()
278 maMatrixRangeList
.Append(aScRange
);
280 ScDocumentImport
& rDoc
= rImport
.GetDoc();
281 boost::scoped_ptr
<ScTokenArray
> pCode(new ScTokenArray
);
282 pCode
->AddStringXML( rFormula
);
283 if( (eGrammar
== formula::FormulaGrammar::GRAM_EXTERNAL
) && !rFormulaNmsp
.isEmpty() )
284 pCode
->AddStringXML( rFormulaNmsp
);
285 rDoc
.setMatrixCells(aScRange
, *pCode
, eGrammar
);
286 rDoc
.getDoc().IncXMLImportedFormulaCount( rFormula
.getLength() );
289 bool ScMyTables::IsPartOfMatrix(const ScAddress
& rScAddress
) const
291 if (!maMatrixRangeList
.empty())
292 return maMatrixRangeList
.In(rScAddress
);
296 /* vim:set shiftwidth=4 softtabstop=4 expandtab: */