1 /**************************************************************
2  *
3  * Licensed to the Apache Software Foundation (ASF) under one
4  * or more contributor license agreements.  See the NOTICE file
5  * distributed with this work for additional information
6  * regarding copyright ownership.  The ASF licenses this file
7  * to you under the Apache License, Version 2.0 (the
8  * "License"); you may not use this file except in compliance
9  * with the License.  You may obtain a copy of the License at
10  *
11  *   http://www.apache.org/licenses/LICENSE-2.0
12  *
13  * Unless required by applicable law or agreed to in writing,
14  * software distributed under the License is distributed on an
15  * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
16  * KIND, either express or implied.  See the License for the
17  * specific language governing permissions and limitations
18  * under the License.
19  *
20  *************************************************************/
21 
22 
23 
24 #include "precompiled_chart2.hxx"
25 
26 #include "ChartModelClone.hxx"
27 #include "ChartModelHelper.hxx"
28 #include "ControllerLockGuard.hxx"
29 #include "DataSourceHelper.hxx"
30 
31 /** === begin UNO includes === **/
32 #include <com/sun/star/chart2/XAnyDescriptionAccess.hpp>
33 #include <com/sun/star/util/XCloneable.hpp>
34 #include <com/sun/star/chart2/XChartDocument.hpp>
35 #include <com/sun/star/view/XSelectionSupplier.hpp>
36 #include <com/sun/star/lang/XComponent.hpp>
37 #include <com/sun/star/chart2/XTitled.hpp>
38 #include <com/sun/star/util/XModifiable.hpp>
39 #include <com/sun/star/chart2/data/XDataSource.hpp>
40 #include <com/sun/star/chart2/data/XLabeledDataSequence.hpp>
41 /** === end UNO includes === **/
42 
43 #include <comphelper/property.hxx>
44 #include <tools/diagnose_ex.h>
45 
46 //......................................................................................................................
47 namespace chart
48 {
49 //......................................................................................................................
50 
51 	/** === begin UNO using === **/
52 	using ::com::sun::star::uno::Reference;
53 	using ::com::sun::star::uno::XInterface;
54 	using ::com::sun::star::uno::UNO_QUERY;
55 	using ::com::sun::star::uno::UNO_QUERY_THROW;
56 	using ::com::sun::star::uno::UNO_SET_THROW;
57 	using ::com::sun::star::uno::Exception;
58 	using ::com::sun::star::uno::RuntimeException;
59 	using ::com::sun::star::uno::Any;
60 	using ::com::sun::star::uno::makeAny;
61 	using ::com::sun::star::uno::Sequence;
62 	using ::com::sun::star::uno::Type;
63     using ::com::sun::star::frame::XModel;
64     using ::com::sun::star::util::XCloneable;
65     using ::com::sun::star::chart2::XChartDocument;
66     using ::com::sun::star::chart2::XInternalDataProvider;
67     using ::com::sun::star::chart2::XAnyDescriptionAccess;
68     using ::com::sun::star::view::XSelectionSupplier;
69     using ::com::sun::star::lang::XComponent;
70     using ::com::sun::star::chart2::XTitled;
71     using ::com::sun::star::util::XModifiable;
72     using ::com::sun::star::chart2::data::XDataSource;
73     using ::com::sun::star::chart2::data::XLabeledDataSequence;
74     /** === end UNO using === **/
75 
76     // =================================================================================================================
77     // = helper
78     // =================================================================================================================
79     namespace
80     {
81         Reference< XModel > lcl_cloneModel( const Reference< XModel > & xModel )
82         {
83             Reference< XModel > xResult;
84             try
85             {
86                 const Reference< XCloneable > xCloneable( xModel, UNO_QUERY_THROW );
87                 xResult.set( xCloneable->createClone(), UNO_QUERY_THROW );
88             }
89             catch( const Exception& )
90             {
91         	    DBG_UNHANDLED_EXCEPTION();
92             }
93             return xResult;
94         }
95 
96     }
97 
98     // =================================================================================================================
99     // = ChartModelClone
100     // =================================================================================================================
101     // -----------------------------------------------------------------------------------------------------------------
102     ChartModelClone::ChartModelClone( const Reference< XModel >& i_model, const ModelFacet i_facet )
103     {
104         m_xModelClone.set( lcl_cloneModel( i_model ) );
105 
106         try
107         {
108             if ( i_facet == E_MODEL_WITH_DATA )
109             {
110                 const Reference< XChartDocument > xChartDoc( m_xModelClone, UNO_QUERY_THROW );
111                 ENSURE_OR_THROW( xChartDoc->hasInternalDataProvider(), "invalid chart model" );
112 
113                 const Reference< XCloneable > xCloneable( xChartDoc->getDataProvider(), UNO_QUERY_THROW );
114                 m_xDataClone.set( xCloneable->createClone(), UNO_QUERY_THROW );
115             }
116 
117             if ( i_facet == E_MODEL_WITH_SELECTION )
118             {
119                 const Reference< XSelectionSupplier > xSelSupp( m_xModelClone->getCurrentController(), UNO_QUERY_THROW );
120                 m_aSelection = xSelSupp->getSelection();
121             }
122         }
123         catch( const Exception& )
124         {
125             DBG_UNHANDLED_EXCEPTION();
126         }
127     }
128 
129 
130     // -----------------------------------------------------------------------------------------------------------------
131     ChartModelClone::~ChartModelClone()
132     {
133         if ( !impl_isDisposed() )
134             dispose();
135     }
136 
137     // -----------------------------------------------------------------------------------------------------------------
138     void ChartModelClone::dispose()
139     {
140         if ( impl_isDisposed() )
141             return;
142 
143         try
144         {
145             Reference< XComponent > xComp( m_xModelClone, UNO_QUERY_THROW );
146             xComp->dispose();
147         }
148         catch( const Exception& )
149         {
150     	    DBG_UNHANDLED_EXCEPTION();
151         }
152         m_xModelClone.clear();
153         m_xDataClone.clear();
154         m_aSelection.clear();
155     }
156 
157     // -----------------------------------------------------------------------------------------------------------------
158     ModelFacet ChartModelClone::getFacet() const
159     {
160         if ( m_aSelection.hasValue() )
161             return E_MODEL_WITH_SELECTION;
162         if ( m_xDataClone.is() )
163             return E_MODEL_WITH_DATA;
164         return E_MODEL;
165     }
166 
167     // -----------------------------------------------------------------------------------------------------------------
168     void ChartModelClone::applyToModel( const Reference< XModel >& i_model ) const
169     {
170         applyModelContentToModel( i_model, m_xModelClone, m_xDataClone );
171 
172         if ( m_aSelection.hasValue() )
173         {
174             try
175             {
176                 Reference< XSelectionSupplier > xCurrentSelectionSuppl( i_model->getCurrentController(), UNO_QUERY_THROW );
177                 xCurrentSelectionSuppl->select( m_aSelection );
178             }
179             catch( const Exception& )
180             {
181         	    DBG_UNHANDLED_EXCEPTION();
182             }
183         }
184     }
185 
186     // -----------------------------------------------------------------------------------------------------------------
187     namespace
188     {
189         void ImplApplyDataToModel( const Reference< XModel >& i_model, const Reference< XInternalDataProvider > & i_data )
190         {
191             Reference< XChartDocument > xDoc( i_model, UNO_QUERY );
192             OSL_ASSERT( xDoc.is() && xDoc->hasInternalDataProvider() );
193 
194             // copy data from stored internal data provider
195             if( xDoc.is() && xDoc->hasInternalDataProvider())
196             {
197                 Reference< XAnyDescriptionAccess > xCurrentData( xDoc->getDataProvider(), UNO_QUERY );
198                 Reference< XAnyDescriptionAccess > xSavedData( i_data, UNO_QUERY );
199                 if ( xCurrentData.is() && xSavedData.is() )
200                 {
201                     xCurrentData->setData( xSavedData->getData() );
202                     xCurrentData->setAnyRowDescriptions( xSavedData->getAnyRowDescriptions());
203                     xCurrentData->setAnyColumnDescriptions( xSavedData->getAnyColumnDescriptions());
204                 }
205             }
206         }
207     }
208 
209     // -----------------------------------------------------------------------------------------------------------------
210     void ChartModelClone::applyModelContentToModel( const Reference< XModel >& i_model,
211         const Reference< XModel >& i_modelToCopyFrom, const Reference< XInternalDataProvider >& i_data )
212     {
213         ENSURE_OR_RETURN_VOID( i_model.is(), "ChartModelElement::applyModelContentToModel: invalid source model!" );
214         ENSURE_OR_RETURN_VOID( i_modelToCopyFrom.is(), "ChartModelElement::applyModelContentToModel: invalid source model!" );
215         try
216         {
217             // /-- loccked controllers of destination
218             ControllerLockGuard aLockedControllers( i_model );
219             Reference< XChartDocument > xSource( i_modelToCopyFrom, UNO_QUERY_THROW );
220             Reference< XChartDocument > xDestination( i_model, UNO_QUERY_THROW );
221 
222             // propagate the correct flag for plotting of hidden values to the data provider and all used sequences
223             ChartModelHelper::setIncludeHiddenCells( ChartModelHelper::isIncludeHiddenCells( i_modelToCopyFrom ) , i_model );
224 
225             // diagram
226             xDestination->setFirstDiagram( xSource->getFirstDiagram() );
227 
228             // main title
229             Reference< XTitled > xDestinationTitled( xDestination, UNO_QUERY_THROW );
230             Reference< XTitled > xSourceTitled( xSource, UNO_QUERY_THROW );
231             xDestinationTitled->setTitleObject( xSourceTitled->getTitleObject() );
232 
233             // page background
234             ::comphelper::copyProperties(
235                 xSource->getPageBackground(),
236                 xDestination->getPageBackground() );
237 
238             // apply data (not applied in standard Undo)
239             if ( i_data.is() )
240                 ImplApplyDataToModel( i_model, i_data );
241 
242             // register all sequences at the internal data provider to get adapted
243             // indexes when columns are added/removed
244             if ( xDestination->hasInternalDataProvider() )
245             {
246                 Reference< XInternalDataProvider > xNewDataProvider( xDestination->getDataProvider(), UNO_QUERY );
247                 Reference< XDataSource > xUsedData( DataSourceHelper::getUsedData( i_model ) );
248                 if ( xUsedData.is() && xNewDataProvider.is() )
249                 {
250                     Sequence< Reference< XLabeledDataSequence > > aData( xUsedData->getDataSequences() );
251                     for( sal_Int32 i=0; i<aData.getLength(); ++i )
252                     {
253                         xNewDataProvider->registerDataSequenceForChanges( aData[i]->getValues() );
254                         xNewDataProvider->registerDataSequenceForChanges( aData[i]->getLabel() );
255                     }
256                 }
257             }
258 
259             // restore modify status
260             Reference< XModifiable > xSourceMod( xSource, UNO_QUERY );
261             Reference< XModifiable > xDestMod( xDestination, UNO_QUERY );
262             if ( xSourceMod.is() && xDestMod.is() && !xSourceMod->isModified() )
263             {
264                 xDestMod->setModified( sal_False );
265             }
266             // \-- loccked controllers of destination
267         }
268         catch( const Exception& )
269         {
270             DBG_UNHANDLED_EXCEPTION();
271         }
272     }
273 
274 
275 //......................................................................................................................
276 } // namespace chart
277 //......................................................................................................................
278