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 // MARKER(update_precomp.py): autogen include statement, do not remove
25 #include "precompiled_chart2.hxx"
26 
27 #include "DialogModel.hxx"
28 #include "RangeSelectionHelper.hxx"
29 #include "PropertyHelper.hxx"
30 #include "DataSeriesHelper.hxx"
31 #include "DataSourceHelper.hxx"
32 #include "DiagramHelper.hxx"
33 #include "macros.hxx"
34 #include "Strings.hrc"
35 #include "ResId.hxx"
36 #include "ContainerHelper.hxx"
37 #include "CommonFunctors.hxx"
38 #include "ControllerLockGuard.hxx"
39 #include "ChartTypeHelper.hxx"
40 #include "ThreeDHelper.hxx"
41 
42 #include <com/sun/star/util/XCloneable.hpp>
43 #include <com/sun/star/chart2/AxisType.hpp>
44 #include <com/sun/star/chart2/XTitled.hpp>
45 #include <com/sun/star/chart2/XCoordinateSystemContainer.hpp>
46 #include <com/sun/star/chart2/XChartTypeContainer.hpp>
47 #include <com/sun/star/chart2/XDataSeriesContainer.hpp>
48 #include <com/sun/star/chart2/data/XDataSink.hpp>
49 
50 #include <tools/string.hxx>
51 
52 #include <utility>
53 #include <algorithm>
54 #include <iterator>
55 #include <functional>
56 #include <numeric>
57 
58 using namespace ::com::sun::star;
59 using namespace ::com::sun::star::chart2;
60 using namespace ::chart::ContainerHelper;
61 
62 using ::com::sun::star::uno::Reference;
63 using ::com::sun::star::uno::Sequence;
64 using ::rtl::OUString;
65 
66 // ----------------------------------------
67 
68 namespace
69 {
70 const OUString lcl_aLabelRole( RTL_CONSTASCII_USTRINGPARAM( "label" ));
71 
72 struct lcl_ChartTypeToSeriesCnt : ::std::unary_function<
73         Reference< XChartType >, Reference< XDataSeriesContainer > >
74 {
operator ()__anon2c8f93a80111::lcl_ChartTypeToSeriesCnt75     Reference< XDataSeriesContainer > operator() (
76         const Reference< XChartType > & xChartType )
77     {
78         return Reference< XDataSeriesContainer >::query( xChartType );
79     }
80 };
81 
lcl_ConvertRole(const OUString & rRoleString,bool bFromInternalToUI)82 OUString lcl_ConvertRole( const OUString & rRoleString, bool bFromInternalToUI )
83 {
84     OUString aResult( rRoleString );
85 
86     typedef ::std::map< OUString, OUString > tTranslationMap;
87     static tTranslationMap aTranslationMap;
88 
89     if( aTranslationMap.size() == 0 )
90     {
91         aTranslationMap[ C2U( "categories" ) ] =   OUString( String( ::chart::SchResId( STR_DATA_ROLE_CATEGORIES )));
92         aTranslationMap[ C2U( "error-bars-x" ) ] = OUString( String( ::chart::SchResId( STR_DATA_ROLE_X_ERROR )));
93         aTranslationMap[ C2U( "error-bars-x-positive" ) ] = OUString( String( ::chart::SchResId( STR_DATA_ROLE_X_ERROR_POSITIVE )));
94         aTranslationMap[ C2U( "error-bars-x-negative" ) ] = OUString( String( ::chart::SchResId( STR_DATA_ROLE_X_ERROR_NEGATIVE )));
95         aTranslationMap[ C2U( "error-bars-y" ) ] = OUString( String( ::chart::SchResId( STR_DATA_ROLE_Y_ERROR )));
96         aTranslationMap[ C2U( "error-bars-y-positive" ) ] = OUString( String( ::chart::SchResId( STR_DATA_ROLE_Y_ERROR_POSITIVE )));
97         aTranslationMap[ C2U( "error-bars-y-negative" ) ] = OUString( String( ::chart::SchResId( STR_DATA_ROLE_Y_ERROR_NEGATIVE )));
98         aTranslationMap[ C2U( "label" ) ] =        OUString( String( ::chart::SchResId( STR_DATA_ROLE_LABEL )));
99         aTranslationMap[ C2U( "values-first" ) ] = OUString( String( ::chart::SchResId( STR_DATA_ROLE_FIRST )));
100         aTranslationMap[ C2U( "values-last" ) ] =  OUString( String( ::chart::SchResId( STR_DATA_ROLE_LAST )));
101         aTranslationMap[ C2U( "values-max" ) ] =   OUString( String( ::chart::SchResId( STR_DATA_ROLE_MAX )));
102         aTranslationMap[ C2U( "values-min" ) ] =   OUString( String( ::chart::SchResId( STR_DATA_ROLE_MIN )));
103         aTranslationMap[ C2U( "values-x" ) ] =     OUString( String( ::chart::SchResId( STR_DATA_ROLE_X )));
104         aTranslationMap[ C2U( "values-y" ) ] =     OUString( String( ::chart::SchResId( STR_DATA_ROLE_Y )));
105         aTranslationMap[ C2U( "values-size" ) ] =  OUString( String( ::chart::SchResId( STR_DATA_ROLE_SIZE )));
106     }
107 
108     if( bFromInternalToUI )
109     {
110         tTranslationMap::const_iterator aIt( aTranslationMap.find( rRoleString ));
111         if( aIt != aTranslationMap.end())
112         {
113             aResult = (*aIt).second;
114         }
115     }
116     else
117     {
118         tTranslationMap::const_iterator aIt(
119             ::std::find_if( aTranslationMap.begin(), aTranslationMap.end(),
120                             ::std::compose1( ::std::bind2nd(
121                                                  ::std::equal_to< tTranslationMap::mapped_type >(),
122                                                  rRoleString ),
123                                              ::std::select2nd< tTranslationMap::value_type >())));
124 
125         if( aIt != aTranslationMap.end())
126             aResult = (*aIt).first;
127     }
128 
129     return aResult;
130 }
131 
132 typedef ::std::map< ::rtl::OUString, sal_Int32 > lcl_tRoleIndexMap;
133 
lcl_createRoleIndexMap(lcl_tRoleIndexMap & rOutMap)134 void lcl_createRoleIndexMap( lcl_tRoleIndexMap & rOutMap )
135 {
136     rOutMap.clear();
137     sal_Int32 nIndex = 0;
138 
139     rOutMap[ C2U( "label" ) ] =                 ++nIndex;
140     rOutMap[ C2U( "categories" ) ] =            ++nIndex;
141     rOutMap[ C2U( "values-x" ) ] =              ++nIndex;
142     rOutMap[ C2U( "values-y" ) ] =              ++nIndex;
143     rOutMap[ C2U( "error-bars-x" ) ] =          ++nIndex;
144     rOutMap[ C2U( "error-bars-x-positive" ) ] = ++nIndex;
145     rOutMap[ C2U( "error-bars-x-negative" ) ] = ++nIndex;
146     rOutMap[ C2U( "error-bars-y" ) ] =          ++nIndex;
147     rOutMap[ C2U( "error-bars-y-positive" ) ] = ++nIndex;
148     rOutMap[ C2U( "error-bars-y-negative" ) ] = ++nIndex;
149     rOutMap[ C2U( "values-first" ) ] =          ++nIndex;
150     rOutMap[ C2U( "values-min" ) ] =            ++nIndex;
151     rOutMap[ C2U( "values-max" ) ] =            ++nIndex;
152     rOutMap[ C2U( "values-last" ) ] =           ++nIndex;
153     rOutMap[ C2U( "values-size" ) ] =           ++nIndex;
154 }
155 
156 struct lcl_DataSeriesContainerAppend : public
157     ::std::iterator< ::std::output_iterator_tag, Reference< XDataSeriesContainer > >
158 {
159     typedef ::std::vector< ::chart::DialogModel::tSeriesWithChartTypeByName > tContainerType;
160 
lcl_DataSeriesContainerAppend__anon2c8f93a80111::lcl_DataSeriesContainerAppend161     explicit lcl_DataSeriesContainerAppend( tContainerType & rCnt )
162             : m_rDestCnt( rCnt )
163     {}
164 
operator =__anon2c8f93a80111::lcl_DataSeriesContainerAppend165     lcl_DataSeriesContainerAppend & operator= ( const value_type & xVal )
166     {
167         try
168         {
169             if( xVal.is())
170             {
171                 Sequence< Reference< XDataSeries > > aSeq( xVal->getDataSeries());
172                 OUString aRole( RTL_CONSTASCII_USTRINGPARAM("values-y"));
173                 Reference< XChartType > xCT( xVal, uno::UNO_QUERY );
174                 if( xCT.is())
175                     aRole = xCT->getRoleOfSequenceForSeriesLabel();
176                 for( sal_Int32 nI = 0; nI < aSeq.getLength(); ++ nI )
177                 {
178                     m_rDestCnt.push_back(
179                         ::chart::DialogModel::tSeriesWithChartTypeByName(
180                             ::chart::DataSeriesHelper::getDataSeriesLabel( aSeq[nI], aRole ),
181                             ::std::make_pair( aSeq[nI], xCT )));
182                 }
183             }
184         }
185         catch( uno::Exception & ex )
186         {
187             ASSERT_EXCEPTION( ex );
188         }
189         return *this;
190     }
191 
operator *__anon2c8f93a80111::lcl_DataSeriesContainerAppend192     lcl_DataSeriesContainerAppend & operator* ()     { return *this; }
operator ++__anon2c8f93a80111::lcl_DataSeriesContainerAppend193     lcl_DataSeriesContainerAppend & operator++ ()    { return *this; }
operator ++__anon2c8f93a80111::lcl_DataSeriesContainerAppend194     lcl_DataSeriesContainerAppend & operator++ (int) { return *this; }
195 
196 private:
197     tContainerType & m_rDestCnt;
198 };
199 
200 struct lcl_RolesWithRangeAppend : public
201     ::std::iterator< ::std::output_iterator_tag, Reference< data::XLabeledDataSequence > >
202 {
203     typedef ::chart::DialogModel::tRolesWithRanges tContainerType;
204 
lcl_RolesWithRangeAppend__anon2c8f93a80111::lcl_RolesWithRangeAppend205     explicit lcl_RolesWithRangeAppend( tContainerType & rCnt,
206                                        const ::rtl::OUString & aLabelRole )
207             : m_rDestCnt( rCnt ),
208               m_aRoleForLabelSeq( aLabelRole )
209     {}
210 
operator =__anon2c8f93a80111::lcl_RolesWithRangeAppend211     lcl_RolesWithRangeAppend & operator= ( const value_type & xVal )
212     {
213         try
214         {
215             if( xVal.is())
216             {
217                 // data sequence
218                 Reference< data::XDataSequence > xSeq( xVal->getValues());
219                 if( xSeq.is())
220                 {
221                     OUString aRole;
222                     Reference< beans::XPropertySet > xProp( xSeq, uno::UNO_QUERY_THROW );
223                     if( xProp->getPropertyValue( C2U("Role")) >>= aRole )
224                     {
225                         m_rDestCnt.insert(
226                             tContainerType::value_type(
227                                 aRole, xSeq->getSourceRangeRepresentation()));
228                         // label
229                         if( aRole.equals( m_aRoleForLabelSeq ))
230                         {
231                             Reference< data::XDataSequence > xLabelSeq( xVal->getLabel());
232                             if( xLabelSeq.is())
233                             {
234                                 m_rDestCnt.insert(
235                                     tContainerType::value_type(
236                                         lcl_aLabelRole, xLabelSeq->getSourceRangeRepresentation()));
237                             }
238                         }
239                     }
240                 }
241             }
242         }
243         catch( uno::Exception & ex )
244         {
245             ASSERT_EXCEPTION( ex );
246         }
247         return *this;
248     }
249 
operator *__anon2c8f93a80111::lcl_RolesWithRangeAppend250     lcl_RolesWithRangeAppend & operator* ()     { return *this; }
operator ++__anon2c8f93a80111::lcl_RolesWithRangeAppend251     lcl_RolesWithRangeAppend & operator++ ()    { return *this; }
operator ++__anon2c8f93a80111::lcl_RolesWithRangeAppend252     lcl_RolesWithRangeAppend & operator++ (int) { return *this; }
253 
254 private:
255     tContainerType & m_rDestCnt;
256     OUString m_aRoleForLabelSeq;
257 };
258 
lcl_SetSequenceRole(const Reference<data::XDataSequence> & xSeq,const OUString & rRole)259 void lcl_SetSequenceRole(
260     const Reference< data::XDataSequence > & xSeq,
261     const OUString & rRole )
262 {
263     Reference< beans::XPropertySet > xProp( xSeq, uno::UNO_QUERY );
264     if( xProp.is())
265         xProp->setPropertyValue( C2U("Role"), uno::makeAny( rRole ));
266 }
267 
lcl_CreateNewSeries(const Reference<uno::XComponentContext> & xContext,const Reference<XChartType> & xChartType,sal_Int32 nNewSeriesIndex,sal_Int32 nTotalNumberOfSeriesInCTGroup,const Reference<XDiagram> & xDiagram,const Reference<XChartTypeTemplate> & xTemplate,bool bCreateDataCachedSequences)268 Reference< XDataSeries > lcl_CreateNewSeries(
269     const Reference< uno::XComponentContext > & xContext,
270     const Reference< XChartType > & xChartType,
271     sal_Int32 nNewSeriesIndex,
272     sal_Int32 nTotalNumberOfSeriesInCTGroup,
273     const Reference< XDiagram > & xDiagram,
274     const Reference< XChartTypeTemplate > & xTemplate,
275     bool bCreateDataCachedSequences )
276 {
277     // create plain series
278     Reference< XDataSeries > xResult(
279         xContext->getServiceManager()->createInstanceWithContext(
280             C2U( "com.sun.star.chart2.DataSeries" ),
281             xContext ), uno::UNO_QUERY );
282     if( xTemplate.is())
283     {
284         Reference< beans::XPropertySet > xResultProp( xResult, uno::UNO_QUERY );
285         if( xResultProp.is())
286         {
287             // @deprecated: correct default color should be found by view
288             // without setting it as hard attribute
289             Reference< XColorScheme > xColorScheme( xDiagram->getDefaultColorScheme());
290             if( xColorScheme.is())
291                 xResultProp->setPropertyValue(
292                     C2U("Color"), uno::makeAny( xColorScheme->getColorByIndex( nNewSeriesIndex )));
293         }
294         sal_Int32 nGroupIndex=0;
295         if( xChartType.is())
296         {
297             Sequence< Reference< XChartType > > aCTs(
298                 ::chart::DiagramHelper::getChartTypesFromDiagram( xDiagram ));
299             for( ; nGroupIndex<aCTs.getLength(); ++nGroupIndex)
300                 if( aCTs[nGroupIndex] == xChartType )
301                     break;
302             if( nGroupIndex == aCTs.getLength())
303                 nGroupIndex = 0;
304         }
305         xTemplate->applyStyle( xResult, nGroupIndex, nNewSeriesIndex, nTotalNumberOfSeriesInCTGroup );
306     }
307 
308     if( bCreateDataCachedSequences )
309     {
310         // set chart type specific roles
311         Reference< data::XDataSink > xSink( xResult, uno::UNO_QUERY );
312         if( xChartType.is() && xSink.is())
313         {
314             ::std::vector< Reference< data::XLabeledDataSequence > > aNewSequences;
315             const OUString aRoleOfSeqForSeriesLabel = xChartType->getRoleOfSequenceForSeriesLabel();
316             const OUString aLabel( String( ::chart::SchResId( STR_DATA_UNNAMED_SERIES )));
317             const Sequence< OUString > aRoles( xChartType->getSupportedMandatoryRoles());
318             const Sequence< OUString > aOptRoles( xChartType->getSupportedOptionalRoles());
319             sal_Int32 nI = 0;
320 
321             for(nI=0; nI<aRoles.getLength(); ++nI)
322             {
323                 if( aRoles[nI].equals( lcl_aLabelRole ))
324                     continue;
325                 Reference< data::XDataSequence > xSeq( ::chart::DataSourceHelper::createCachedDataSequence() );
326                 lcl_SetSequenceRole( xSeq, aRoles[nI] );
327                 // assert that aRoleOfSeqForSeriesLabel is part of the mandatory roles
328                 if( aRoles[nI].equals( aRoleOfSeqForSeriesLabel ))
329                 {
330                     Reference< data::XDataSequence > xLabel( ::chart::DataSourceHelper::createCachedDataSequence( aLabel ));
331                     lcl_SetSequenceRole( xLabel, lcl_aLabelRole );
332                     aNewSequences.push_back( ::chart::DataSourceHelper::createLabeledDataSequence( xSeq, xLabel ));
333                 }
334                 else
335                     aNewSequences.push_back( ::chart::DataSourceHelper::createLabeledDataSequence( xSeq ));
336             }
337 
338             for(nI=0; nI<aOptRoles.getLength(); ++nI)
339             {
340                 if( aOptRoles[nI].equals( lcl_aLabelRole ))
341                     continue;
342                 Reference< data::XDataSequence > xSeq( ::chart::DataSourceHelper::createCachedDataSequence());
343                 lcl_SetSequenceRole( xSeq, aOptRoles[nI] );
344                 aNewSequences.push_back( ::chart::DataSourceHelper::createLabeledDataSequence( xSeq ));
345             }
346 
347             xSink->setData( ContainerToSequence( aNewSequences ));
348         }
349     }
350 
351     return xResult;
352 }
353 
354 struct lcl_addSeriesNumber : public ::std::binary_function<
355         sal_Int32, Reference< XDataSeriesContainer >, sal_Int32 >
356 {
operator ()__anon2c8f93a80111::lcl_addSeriesNumber357     sal_Int32 operator() ( sal_Int32 nCurrentNumber, const Reference< XDataSeriesContainer > & xCnt ) const
358     {
359         if( xCnt.is())
360             return nCurrentNumber + (xCnt->getDataSeries().getLength());
361         return nCurrentNumber;
362     }
363 };
364 
365 } // anonymous namespace
366 
367 // ----------------------------------------
368 
369 
370 namespace chart
371 {
372 
DialogModel(const Reference<XChartDocument> & xChartDocument,const Reference<uno::XComponentContext> & xContext)373 DialogModel::DialogModel(
374     const Reference< XChartDocument > & xChartDocument,
375     const Reference< uno::XComponentContext > & xContext ) :
376         m_xChartDocument( xChartDocument ),
377         m_xContext( xContext ),
378         m_aTimerTriggeredControllerLock( uno::Reference< frame::XModel >( m_xChartDocument, uno::UNO_QUERY ) )
379 {
380 }
381 
~DialogModel()382 DialogModel::~DialogModel()
383 {}
384 
setTemplate(const Reference<XChartTypeTemplate> & xTemplate)385 void DialogModel::setTemplate(
386     const Reference< XChartTypeTemplate > & xTemplate )
387 {
388     m_xTemplate = xTemplate;
389 }
390 
391 ::boost::shared_ptr< RangeSelectionHelper >
getRangeSelectionHelper() const392     DialogModel::getRangeSelectionHelper() const
393 {
394     if( ! m_spRangeSelectionHelper.get())
395         m_spRangeSelectionHelper.reset(
396             new RangeSelectionHelper( m_xChartDocument ));
397 
398     return m_spRangeSelectionHelper;
399 }
400 
getChartModel() const401 Reference< frame::XModel > DialogModel::getChartModel() const
402 {
403     Reference< frame::XModel > xResult( m_xChartDocument, uno::UNO_QUERY );
404     return xResult;
405 }
406 
getDataProvider() const407 Reference< data::XDataProvider > DialogModel::getDataProvider() const
408 {
409     Reference< data::XDataProvider > xResult;
410     if( m_xChartDocument.is())
411         xResult.set( m_xChartDocument->getDataProvider());
412     return xResult;
413 }
414 
415 ::std::vector< Reference< XDataSeriesContainer > >
getAllDataSeriesContainers() const416     DialogModel::getAllDataSeriesContainers() const
417 {
418     ::std::vector< Reference< XDataSeriesContainer > > aResult;
419 
420     try
421     {
422         Reference< XDiagram > xDiagram;
423         if( m_xChartDocument.is())
424             xDiagram.set( m_xChartDocument->getFirstDiagram());
425         if( xDiagram.is())
426         {
427             Reference< XCoordinateSystemContainer > xCooSysCnt(
428                 xDiagram, uno::UNO_QUERY_THROW );
429             Sequence< Reference< XCoordinateSystem > > aCooSysSeq(
430                 xCooSysCnt->getCoordinateSystems());
431             for( sal_Int32 i=0; i<aCooSysSeq.getLength(); ++i )
432             {
433                 Reference< XChartTypeContainer > xCTCnt( aCooSysSeq[i], uno::UNO_QUERY_THROW );
434                 Sequence< Reference< XChartType > > aChartTypeSeq( xCTCnt->getChartTypes());
435                 ::std::transform(
436                     aChartTypeSeq.getConstArray(), aChartTypeSeq.getConstArray() + aChartTypeSeq.getLength(),
437                     ::std::back_inserter( aResult ),
438                     lcl_ChartTypeToSeriesCnt() );
439             }
440         }
441     }
442     catch( uno::Exception & ex )
443     {
444         ASSERT_EXCEPTION( ex );
445     }
446 
447     return aResult;
448 }
449 
450 ::std::vector< DialogModel::tSeriesWithChartTypeByName >
getAllDataSeriesWithLabel() const451     DialogModel::getAllDataSeriesWithLabel() const
452 {
453     ::std::vector< tSeriesWithChartTypeByName > aResult;
454     ::std::vector< Reference< XDataSeriesContainer > > aContainers(
455         getAllDataSeriesContainers());
456 
457     ::std::copy( aContainers.begin(), aContainers.end(),
458                  lcl_DataSeriesContainerAppend( aResult ));
459     return aResult;
460 }
461 
getRolesWithRanges(const Reference<XDataSeries> & xSeries,const::rtl::OUString & aRoleOfSequenceForLabel,const Reference<chart2::XChartType> & xChartType) const462 DialogModel::tRolesWithRanges DialogModel::getRolesWithRanges(
463     const Reference< XDataSeries > & xSeries,
464     const ::rtl::OUString & aRoleOfSequenceForLabel,
465     const Reference< chart2::XChartType > & xChartType ) const
466 {
467     DialogModel::tRolesWithRanges aResult;
468     try
469     {
470         Reference< data::XDataSource > xSource( xSeries, uno::UNO_QUERY_THROW );
471         const Sequence< Reference< data::XLabeledDataSequence > > aSeq( xSource->getDataSequences());
472         ::std::copy( aSeq.getConstArray(), aSeq.getConstArray() + aSeq.getLength(),
473                      lcl_RolesWithRangeAppend( aResult, aRoleOfSequenceForLabel ));
474         if( xChartType.is())
475         {
476             // add missing mandatory roles
477             Sequence< OUString > aRoles( xChartType->getSupportedMandatoryRoles());
478             OUString aEmptyString;
479             sal_Int32 nI = 0;
480             for( nI=0; nI < aRoles.getLength(); ++nI )
481             {
482                 if( aResult.find( aRoles[nI] ) == aResult.end() )
483                     aResult.insert( DialogModel::tRolesWithRanges::value_type( aRoles[nI], aEmptyString ));
484             }
485 
486             // add missing optional roles
487             aRoles = xChartType->getSupportedOptionalRoles();
488             for( nI=0; nI < aRoles.getLength(); ++nI )
489             {
490                 if( aResult.find( aRoles[nI] ) == aResult.end() )
491                     aResult.insert( DialogModel::tRolesWithRanges::value_type( aRoles[nI], aEmptyString ));
492             }
493         }
494     }
495     catch( uno::Exception & ex )
496     {
497         ASSERT_EXCEPTION( ex );
498     }
499     return aResult;
500 }
501 
moveSeries(const Reference<XDataSeries> & xSeries,eMoveDirection eDirection)502 void DialogModel::moveSeries(
503     const Reference< XDataSeries > & xSeries,
504     eMoveDirection eDirection )
505 {
506     m_aTimerTriggeredControllerLock.startTimer();
507     ControllerLockGuard aLockedControllers( Reference< frame::XModel >( m_xChartDocument, uno::UNO_QUERY ) );
508 
509     Reference< XDiagram > xDiagram( m_xChartDocument->getFirstDiagram());
510     DiagramHelper::moveSeries( xDiagram, xSeries, eDirection==MOVE_UP );
511 }
512 
insertSeriesAfter(const Reference<XDataSeries> & xSeries,const Reference<XChartType> & xChartType,bool bCreateDataCachedSequences)513 Reference< chart2::XDataSeries > DialogModel::insertSeriesAfter(
514     const Reference< XDataSeries > & xSeries,
515     const Reference< XChartType > & xChartType,
516     bool bCreateDataCachedSequences /* = false */ )
517 {
518     m_aTimerTriggeredControllerLock.startTimer();
519     ControllerLockGuard aLockedControllers( Reference< frame::XModel >( m_xChartDocument, uno::UNO_QUERY ) );
520     Reference< XDataSeries > xNewSeries;
521 
522     try
523     {
524         Reference< chart2::XDiagram > xDiagram( m_xChartDocument->getFirstDiagram() );
525         ThreeDLookScheme e3DScheme = ThreeDHelper::detectScheme( xDiagram );
526 
527         sal_Int32 nSeriesInChartType = 0;
528         const sal_Int32 nTotalSeries = countSeries();
529         if( xChartType.is())
530         {
531             Reference< XDataSeriesContainer > xCnt( xChartType, uno::UNO_QUERY_THROW );
532             nSeriesInChartType = xCnt->getDataSeries().getLength();
533         }
534 
535         // create new series
536         xNewSeries.set(
537             lcl_CreateNewSeries(
538                 m_xContext,
539                 xChartType,
540                 nTotalSeries, // new series' index
541                 nSeriesInChartType,
542                 xDiagram,
543                 m_xTemplate,
544                 bCreateDataCachedSequences ));
545 
546         // add new series to container
547         if( xNewSeries.is())
548         {
549             Reference< XDataSeriesContainer > xSeriesCnt( xChartType, uno::UNO_QUERY_THROW );
550             ::std::vector< Reference< XDataSeries > > aSeries(
551                 SequenceToVector( xSeriesCnt->getDataSeries()));
552             ::std::vector< Reference< XDataSeries > >::iterator aIt =
553                   ::std::find( aSeries.begin(), aSeries.end(), xSeries );
554             if( aIt == aSeries.end())
555                 // if we have no series we insert at the first position.
556                 aIt = aSeries.begin();
557             else
558                 // vector::insert inserts before, so we have to advance
559                 ++aIt;
560             aSeries.insert( aIt, xNewSeries );
561             xSeriesCnt->setDataSeries( ContainerToSequence( aSeries ));
562         }
563 
564         ThreeDHelper::setScheme( xDiagram, e3DScheme );
565     }
566     catch( uno::Exception & ex )
567     {
568         ASSERT_EXCEPTION( ex );
569     }
570     return xNewSeries;
571 }
572 
deleteSeries(const Reference<XDataSeries> & xSeries,const Reference<XChartType> & xChartType)573 void DialogModel::deleteSeries(
574     const Reference< XDataSeries > & xSeries,
575     const Reference< XChartType > & xChartType )
576 {
577     m_aTimerTriggeredControllerLock.startTimer();
578     ControllerLockGuard aLockedControllers( Reference< frame::XModel >( m_xChartDocument, uno::UNO_QUERY ) );
579 
580     DataSeriesHelper::deleteSeries( xSeries, xChartType );
581 }
582 
getCategories() const583 Reference< data::XLabeledDataSequence > DialogModel::getCategories() const
584 {
585     Reference< data::XLabeledDataSequence > xResult;
586     try
587     {
588         if( m_xChartDocument.is())
589         {
590             Reference< chart2::XDiagram > xDiagram( m_xChartDocument->getFirstDiagram());
591             xResult.set( DiagramHelper::getCategoriesFromDiagram( xDiagram ));
592         }
593     }
594     catch( uno::Exception & ex )
595     {
596         ASSERT_EXCEPTION( ex );
597     }
598     return xResult;
599 }
600 
setCategories(const Reference<chart2::data::XLabeledDataSequence> & xCategories)601 void DialogModel::setCategories( const Reference< chart2::data::XLabeledDataSequence > & xCategories )
602 {
603     if( m_xChartDocument.is())
604     {
605         Reference< chart2::XDiagram > xDiagram( m_xChartDocument->getFirstDiagram());
606         if( xDiagram.is())
607         {
608             // categories
609             bool bSupportsCategories = true;
610 
611             Reference< XChartType > xFirstChartType( DiagramHelper::getChartTypeByIndex( xDiagram, 0 ) );
612             if( xFirstChartType.is() )
613             {
614                 sal_Int32 nAxisType = ChartTypeHelper::getAxisType( xFirstChartType, 0 ); // x-axis
615                 bSupportsCategories = (nAxisType == AxisType::CATEGORY);
616             }
617             DiagramHelper::setCategoriesToDiagram( xCategories, xDiagram, true, bSupportsCategories );
618         }
619     }
620 }
621 
getCategoriesRange() const622 OUString DialogModel::getCategoriesRange() const
623 {
624     Reference< data::XLabeledDataSequence > xLSeq( getCategories());
625     OUString aRange;
626     if( xLSeq.is())
627     {
628         Reference< data::XDataSequence > xSeq( xLSeq->getValues());
629         if( xSeq.is())
630             aRange = xSeq->getSourceRangeRepresentation();
631     }
632     return aRange;
633 }
634 
isCategoryDiagram() const635 bool DialogModel::isCategoryDiagram() const
636 {
637     bool bRet = false;
638     if( m_xChartDocument.is())
639         bRet = DiagramHelper::isCategoryDiagram( m_xChartDocument->getFirstDiagram() );
640     return bRet;
641 }
642 
detectArguments(OUString & rOutRangeString,bool & rOutUseColumns,bool & rOutFirstCellAsLabel,bool & rOutHasCategories) const643 void DialogModel::detectArguments(
644     OUString & rOutRangeString,
645     bool & rOutUseColumns,
646     bool & rOutFirstCellAsLabel,
647     bool & rOutHasCategories ) const
648 {
649     try
650     {
651         uno::Sequence< sal_Int32 > aSequenceMapping;//todo YYYX
652 
653         // Note: unused data is currently not supported in being passed to detectRangeSegmentation
654         if( m_xChartDocument.is())
655             DataSourceHelper::detectRangeSegmentation(
656                 Reference< frame::XModel >( m_xChartDocument, uno::UNO_QUERY_THROW ),
657                 rOutRangeString, aSequenceMapping, rOutUseColumns, rOutFirstCellAsLabel, rOutHasCategories );
658     }
659     catch( uno::Exception & ex )
660     {
661         ASSERT_EXCEPTION( ex );
662     }
663 }
664 
allArgumentsForRectRangeDetected() const665 bool DialogModel::allArgumentsForRectRangeDetected() const
666 {
667     return DataSourceHelper::allArgumentsForRectRangeDetected( m_xChartDocument );
668 }
669 
startControllerLockTimer()670 void DialogModel::startControllerLockTimer()
671 {
672     m_aTimerTriggeredControllerLock.startTimer();
673 }
674 
setData(const Sequence<beans::PropertyValue> & rArguments)675 bool DialogModel::setData(
676     const Sequence< beans::PropertyValue > & rArguments )
677 {
678     m_aTimerTriggeredControllerLock.startTimer();
679     ControllerLockGuard aLockedControllers( Reference< frame::XModel >( m_xChartDocument, uno::UNO_QUERY ) );
680 
681     Reference< data::XDataProvider > xDataProvider( getDataProvider());
682     if( ! xDataProvider.is() ||
683         ! m_xTemplate.is() )
684     {
685         OSL_ENSURE( false, "Model objects missing" );
686         return false;
687     }
688 
689     try
690     {
691         Reference< chart2::data::XDataSource > xDataSource(
692             xDataProvider->createDataSource( rArguments ) );
693 
694         Reference< chart2::XDataInterpreter > xInterpreter(
695             m_xTemplate->getDataInterpreter());
696         if( xInterpreter.is())
697         {
698             Reference< chart2::XDiagram > xDiagram( m_xChartDocument->getFirstDiagram() );
699             ThreeDLookScheme e3DScheme = ThreeDHelper::detectScheme( xDiagram );
700 
701             ::std::vector< Reference< XDataSeries > > aSeriesToReUse(
702                 DiagramHelper::getDataSeriesFromDiagram( xDiagram ));
703             applyInterpretedData(
704                 xInterpreter->interpretDataSource(
705                     xDataSource, rArguments,
706                     ContainerToSequence( aSeriesToReUse )),
707                 aSeriesToReUse,
708                 true /* bSetStyles */);
709 
710             ThreeDHelper::setScheme( xDiagram, e3DScheme );
711         }
712     }
713     catch( uno::Exception & ex )
714     {
715         ASSERT_EXCEPTION( ex );
716         return false;
717     }
718 
719     return true;
720 }
721 
ConvertRoleFromInternalToUI(const OUString & rRoleString)722 OUString DialogModel::ConvertRoleFromInternalToUI( const OUString & rRoleString )
723 {
724     return lcl_ConvertRole( rRoleString, true );
725 }
726 
GetRoleDataLabel()727 OUString DialogModel::GetRoleDataLabel()
728 {
729     return OUString( String( ::chart::SchResId( STR_OBJECT_DATALABELS )));
730 }
731 
GetRoleIndexForSorting(const::rtl::OUString & rInternalRoleString)732 sal_Int32 DialogModel::GetRoleIndexForSorting( const ::rtl::OUString & rInternalRoleString )
733 {
734     static lcl_tRoleIndexMap aRoleIndexMap;
735 
736     if( aRoleIndexMap.empty())
737         lcl_createRoleIndexMap( aRoleIndexMap );
738 
739     lcl_tRoleIndexMap::const_iterator aIt( aRoleIndexMap.find( rInternalRoleString ));
740     if( aIt != aRoleIndexMap.end())
741         return aIt->second;
742 
743     return 0;
744 }
745 
746 // private methods
747 
applyInterpretedData(const InterpretedData & rNewData,const::std::vector<Reference<XDataSeries>> & rSeriesToReUse,bool bSetStyles)748 void DialogModel::applyInterpretedData(
749     const InterpretedData & rNewData,
750     const ::std::vector< Reference< XDataSeries > > & rSeriesToReUse,
751     bool bSetStyles )
752 {
753     if( ! m_xChartDocument.is())
754         return;
755 
756     m_aTimerTriggeredControllerLock.startTimer();
757     Reference< XDiagram > xDiagram( m_xChartDocument->getFirstDiagram());
758     if( xDiagram.is())
759     {
760         // styles
761         if( bSetStyles && m_xTemplate.is() )
762         {
763             sal_Int32 nGroup = 0;
764             sal_Int32 nSeriesCounter = 0;
765             sal_Int32 nNewSeriesIndex = static_cast< sal_Int32 >( rSeriesToReUse.size());
766             const sal_Int32 nOuterSize=rNewData.Series.getLength();
767 
768             for(; nGroup < nOuterSize; ++nGroup)
769             {
770                 Sequence< Reference< XDataSeries > > aSeries( rNewData.Series[ nGroup ] );
771                 const sal_Int32 nSeriesInGroup = aSeries.getLength();
772                 for( sal_Int32 nSeries=0; nSeries<nSeriesInGroup; ++nSeries, ++nSeriesCounter )
773                 {
774                     if( ::std::find( rSeriesToReUse.begin(), rSeriesToReUse.end(), aSeries[nSeries] )
775                         == rSeriesToReUse.end())
776                     {
777                         Reference< beans::XPropertySet > xSeriesProp( aSeries[nSeries], uno::UNO_QUERY );
778                         if( xSeriesProp.is())
779                         {
780                             // @deprecated: correct default color should be found by view
781                             // without setting it as hard attribute
782                             Reference< XColorScheme > xColorScheme( xDiagram->getDefaultColorScheme());
783                             if( xColorScheme.is())
784                                 xSeriesProp->setPropertyValue(
785                                     C2U("Color"), uno::makeAny( xColorScheme->getColorByIndex( nSeriesCounter )));
786                         }
787                         m_xTemplate->applyStyle( aSeries[nSeries], nGroup, nNewSeriesIndex++, nSeriesInGroup );
788                     }
789                 }
790             }
791         }
792 
793         // data series
794         ::std::vector< Reference< XDataSeriesContainer > > aSeriesCnt( getAllDataSeriesContainers());
795         ::std::vector< Sequence< Reference< XDataSeries > > > aNewSeries(
796             SequenceToVector( rNewData.Series ));
797 
798         OSL_ASSERT( aSeriesCnt.size() == aNewSeries.size());
799 
800         ::std::vector< Sequence< Reference< XDataSeries > > >::const_iterator aSrcIt( aNewSeries.begin());
801         ::std::vector< Reference< XDataSeriesContainer > >::iterator aDestIt( aSeriesCnt.begin());
802         for(; aSrcIt != aNewSeries.end() && aDestIt != aSeriesCnt.end();
803             ++aSrcIt, ++aDestIt )
804         {
805             try
806             {
807                 OSL_ASSERT( (*aDestIt).is());
808                 (*aDestIt)->setDataSeries( *aSrcIt );
809             }
810             catch( uno::Exception & ex )
811             {
812                 ASSERT_EXCEPTION( ex );
813             }
814         }
815 
816         DialogModel::setCategories(rNewData.Categories);
817     }
818 }
819 
countSeries() const820 sal_Int32 DialogModel::countSeries() const
821 {
822     ::std::vector< Reference< XDataSeriesContainer > > aCnt( getAllDataSeriesContainers());
823     return ::std::accumulate( aCnt.begin(), aCnt.end(), 0, lcl_addSeriesNumber());
824 }
825 
826 } //  namespace chart
827