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_chart2.hxx"
26*b1cdbd2cSJim Jagielski 
27*b1cdbd2cSJim Jagielski #include "InternalData.hxx"
28*b1cdbd2cSJim Jagielski #include "ResId.hxx"
29*b1cdbd2cSJim Jagielski #include "Strings.hrc"
30*b1cdbd2cSJim Jagielski #include "macros.hxx"
31*b1cdbd2cSJim Jagielski 
32*b1cdbd2cSJim Jagielski #include <rtl/math.hxx>
33*b1cdbd2cSJim Jagielski #include <algorithm>
34*b1cdbd2cSJim Jagielski 
35*b1cdbd2cSJim Jagielski using ::com::sun::star::uno::Sequence;
36*b1cdbd2cSJim Jagielski using ::rtl::OUString;
37*b1cdbd2cSJim Jagielski 
38*b1cdbd2cSJim Jagielski using namespace ::com::sun::star;
39*b1cdbd2cSJim Jagielski using namespace ::std;
40*b1cdbd2cSJim Jagielski 
41*b1cdbd2cSJim Jagielski namespace chart
42*b1cdbd2cSJim Jagielski {
43*b1cdbd2cSJim Jagielski 
44*b1cdbd2cSJim Jagielski // ----------------------------------------
45*b1cdbd2cSJim Jagielski namespace
46*b1cdbd2cSJim Jagielski {
47*b1cdbd2cSJim Jagielski struct lcl_NumberedStringGenerator
48*b1cdbd2cSJim Jagielski {
lcl_NumberedStringGeneratorchart::__anonfac9888a0111::lcl_NumberedStringGenerator49*b1cdbd2cSJim Jagielski     lcl_NumberedStringGenerator( const OUString & rStub, const OUString & rWildcard ) :
50*b1cdbd2cSJim Jagielski             m_aStub( rStub ),
51*b1cdbd2cSJim Jagielski             m_nCounter( 0 ),
52*b1cdbd2cSJim Jagielski             m_nStubStartIndex( rStub.indexOf( rWildcard )),
53*b1cdbd2cSJim Jagielski             m_nWildcardLength( rWildcard.getLength())
54*b1cdbd2cSJim Jagielski     {
55*b1cdbd2cSJim Jagielski     }
operator ()chart::__anonfac9888a0111::lcl_NumberedStringGenerator56*b1cdbd2cSJim Jagielski     vector< uno::Any > operator()()
57*b1cdbd2cSJim Jagielski     {
58*b1cdbd2cSJim Jagielski         vector< uno::Any > aRet(1);
59*b1cdbd2cSJim Jagielski         aRet[0] = uno::makeAny( m_aStub.replaceAt( m_nStubStartIndex, m_nWildcardLength, OUString::valueOf( ++m_nCounter )) );
60*b1cdbd2cSJim Jagielski         return aRet;
61*b1cdbd2cSJim Jagielski     }
62*b1cdbd2cSJim Jagielski private:
63*b1cdbd2cSJim Jagielski     OUString m_aStub;
64*b1cdbd2cSJim Jagielski     sal_Int32 m_nCounter;
65*b1cdbd2cSJim Jagielski     const sal_Int32 m_nStubStartIndex;
66*b1cdbd2cSJim Jagielski     const sal_Int32 m_nWildcardLength;
67*b1cdbd2cSJim Jagielski };
68*b1cdbd2cSJim Jagielski 
69*b1cdbd2cSJim Jagielski template< typename T >
lcl_ValarrayToSequence(const::std::valarray<T> & rValarray)70*b1cdbd2cSJim Jagielski     Sequence< T > lcl_ValarrayToSequence( const ::std::valarray< T > & rValarray )
71*b1cdbd2cSJim Jagielski {
72*b1cdbd2cSJim Jagielski     // is there a more elegant way of conversion?
73*b1cdbd2cSJim Jagielski     Sequence< T > aResult( rValarray.size());
74*b1cdbd2cSJim Jagielski     for( size_t i = 0; i < rValarray.size(); ++i )
75*b1cdbd2cSJim Jagielski         aResult[i] = rValarray[i];
76*b1cdbd2cSJim Jagielski     return aResult;
77*b1cdbd2cSJim Jagielski }
78*b1cdbd2cSJim Jagielski 
79*b1cdbd2cSJim Jagielski } // anonymous namespace
80*b1cdbd2cSJim Jagielski // ----------------------------------------
81*b1cdbd2cSJim Jagielski 
InternalData()82*b1cdbd2cSJim Jagielski InternalData::InternalData()
83*b1cdbd2cSJim Jagielski     : m_nColumnCount( 0 )
84*b1cdbd2cSJim Jagielski     , m_nRowCount( 0 )
85*b1cdbd2cSJim Jagielski     , m_aRowLabels( 0 )
86*b1cdbd2cSJim Jagielski     , m_aColumnLabels( 0 )
87*b1cdbd2cSJim Jagielski {}
88*b1cdbd2cSJim Jagielski 
createDefaultData()89*b1cdbd2cSJim Jagielski void InternalData::createDefaultData()
90*b1cdbd2cSJim Jagielski {
91*b1cdbd2cSJim Jagielski     const sal_Int32 nRowCount = 4;
92*b1cdbd2cSJim Jagielski     const sal_Int32 nColumnCount = 3;
93*b1cdbd2cSJim Jagielski 
94*b1cdbd2cSJim Jagielski     m_nRowCount = nRowCount;
95*b1cdbd2cSJim Jagielski     m_nColumnCount = nColumnCount;
96*b1cdbd2cSJim Jagielski     const sal_Int32 nSize = nColumnCount * nRowCount;
97*b1cdbd2cSJim Jagielski     // @todo: localize this!
98*b1cdbd2cSJim Jagielski     const OUString aRowName( ::chart::SchResId::getResString( STR_ROW_LABEL ));
99*b1cdbd2cSJim Jagielski     const OUString aColName( ::chart::SchResId::getResString( STR_COLUMN_LABEL ));
100*b1cdbd2cSJim Jagielski 
101*b1cdbd2cSJim Jagielski     const double fDefaultData[ nSize ] =
102*b1cdbd2cSJim Jagielski         { 9.10, 3.20, 4.54,
103*b1cdbd2cSJim Jagielski           2.40, 8.80, 9.65,
104*b1cdbd2cSJim Jagielski           3.10, 1.50, 3.70,
105*b1cdbd2cSJim Jagielski           4.30, 9.02, 6.20 };
106*b1cdbd2cSJim Jagielski 
107*b1cdbd2cSJim Jagielski     m_aData.resize( nSize );
108*b1cdbd2cSJim Jagielski     for( sal_Int32 i=0; i<nSize; ++i )
109*b1cdbd2cSJim Jagielski         m_aData[i] = fDefaultData[i];
110*b1cdbd2cSJim Jagielski 
111*b1cdbd2cSJim Jagielski     m_aRowLabels.clear();
112*b1cdbd2cSJim Jagielski     m_aRowLabels.reserve( m_nRowCount );
113*b1cdbd2cSJim Jagielski     generate_n( back_inserter( m_aRowLabels ), m_nRowCount,
114*b1cdbd2cSJim Jagielski         lcl_NumberedStringGenerator( aRowName, C2U("%ROWNUMBER") ));
115*b1cdbd2cSJim Jagielski 
116*b1cdbd2cSJim Jagielski     m_aColumnLabels.clear();
117*b1cdbd2cSJim Jagielski     m_aColumnLabels.reserve( m_nColumnCount );
118*b1cdbd2cSJim Jagielski     generate_n( back_inserter( m_aColumnLabels ), m_nColumnCount,
119*b1cdbd2cSJim Jagielski         lcl_NumberedStringGenerator( aColName, C2U("%COLUMNNUMBER") ));
120*b1cdbd2cSJim Jagielski }
121*b1cdbd2cSJim Jagielski 
setData(const Sequence<Sequence<double>> & rDataInRows)122*b1cdbd2cSJim Jagielski void InternalData::setData( const Sequence< Sequence< double > >& rDataInRows )
123*b1cdbd2cSJim Jagielski {
124*b1cdbd2cSJim Jagielski     m_nRowCount = rDataInRows.getLength();
125*b1cdbd2cSJim Jagielski     m_nColumnCount = (m_nRowCount ? rDataInRows[0].getLength() : 0);
126*b1cdbd2cSJim Jagielski 
127*b1cdbd2cSJim Jagielski     if( m_aRowLabels.size() != static_cast< sal_uInt32 >( m_nRowCount ))
128*b1cdbd2cSJim Jagielski         m_aRowLabels.resize( m_nRowCount );
129*b1cdbd2cSJim Jagielski     if( m_aColumnLabels.size() != static_cast< sal_uInt32 >( m_nColumnCount ))
130*b1cdbd2cSJim Jagielski         m_aColumnLabels.resize( m_nColumnCount );
131*b1cdbd2cSJim Jagielski 
132*b1cdbd2cSJim Jagielski     m_aData.resize( m_nRowCount * m_nColumnCount );
133*b1cdbd2cSJim Jagielski     double fNan;
134*b1cdbd2cSJim Jagielski     ::rtl::math::setNan( & fNan );
135*b1cdbd2cSJim Jagielski     // set all values to Nan
136*b1cdbd2cSJim Jagielski     m_aData = fNan;
137*b1cdbd2cSJim Jagielski 
138*b1cdbd2cSJim Jagielski     for( sal_Int32 nRow=0; nRow<m_nRowCount; ++nRow )
139*b1cdbd2cSJim Jagielski     {
140*b1cdbd2cSJim Jagielski         int nDataIdx = nRow*m_nColumnCount;
141*b1cdbd2cSJim Jagielski         const sal_Int32 nMax = ::std::min( rDataInRows[nRow].getLength(), m_nColumnCount );
142*b1cdbd2cSJim Jagielski         for( sal_Int32 nCol=0; nCol < nMax; ++nCol )
143*b1cdbd2cSJim Jagielski         {
144*b1cdbd2cSJim Jagielski             m_aData[nDataIdx] = rDataInRows[nRow][nCol];
145*b1cdbd2cSJim Jagielski             nDataIdx += 1;
146*b1cdbd2cSJim Jagielski         }
147*b1cdbd2cSJim Jagielski     }
148*b1cdbd2cSJim Jagielski }
149*b1cdbd2cSJim Jagielski 
getData() const150*b1cdbd2cSJim Jagielski Sequence< Sequence< double > > InternalData::getData() const
151*b1cdbd2cSJim Jagielski {
152*b1cdbd2cSJim Jagielski     Sequence< Sequence< double > > aResult( m_nRowCount );
153*b1cdbd2cSJim Jagielski 
154*b1cdbd2cSJim Jagielski     for( sal_Int32 i=0; i<m_nRowCount; ++i )
155*b1cdbd2cSJim Jagielski         aResult[i] = lcl_ValarrayToSequence< tDataType::value_type >(
156*b1cdbd2cSJim Jagielski             m_aData[ ::std::slice( i*m_nColumnCount, m_nColumnCount, 1 ) ] );
157*b1cdbd2cSJim Jagielski 
158*b1cdbd2cSJim Jagielski     return aResult;
159*b1cdbd2cSJim Jagielski }
160*b1cdbd2cSJim Jagielski 
getColumnValues(sal_Int32 nColumnIndex) const161*b1cdbd2cSJim Jagielski Sequence< double > InternalData::getColumnValues( sal_Int32 nColumnIndex ) const
162*b1cdbd2cSJim Jagielski {
163*b1cdbd2cSJim Jagielski     if( nColumnIndex >= 0 && nColumnIndex < m_nColumnCount )
164*b1cdbd2cSJim Jagielski         return lcl_ValarrayToSequence< tDataType::value_type >(
165*b1cdbd2cSJim Jagielski             m_aData[ ::std::slice( nColumnIndex, m_nRowCount, m_nColumnCount ) ] );
166*b1cdbd2cSJim Jagielski     return Sequence< double >();
167*b1cdbd2cSJim Jagielski }
getRowValues(sal_Int32 nRowIndex) const168*b1cdbd2cSJim Jagielski Sequence< double > InternalData::getRowValues( sal_Int32 nRowIndex ) const
169*b1cdbd2cSJim Jagielski {
170*b1cdbd2cSJim Jagielski     if( nRowIndex >= 0 && nRowIndex < m_nRowCount )
171*b1cdbd2cSJim Jagielski         return lcl_ValarrayToSequence< tDataType::value_type >(
172*b1cdbd2cSJim Jagielski             m_aData[ ::std::slice( nRowIndex*m_nColumnCount, m_nColumnCount, 1 ) ] );
173*b1cdbd2cSJim Jagielski     return Sequence< double >();
174*b1cdbd2cSJim Jagielski }
175*b1cdbd2cSJim Jagielski 
setColumnValues(sal_Int32 nColumnIndex,const vector<double> & rNewData)176*b1cdbd2cSJim Jagielski void InternalData::setColumnValues( sal_Int32 nColumnIndex, const vector< double > & rNewData )
177*b1cdbd2cSJim Jagielski {
178*b1cdbd2cSJim Jagielski     if( nColumnIndex < 0 )
179*b1cdbd2cSJim Jagielski         return;
180*b1cdbd2cSJim Jagielski     enlargeData( nColumnIndex + 1, rNewData.size() );
181*b1cdbd2cSJim Jagielski 
182*b1cdbd2cSJim Jagielski     tDataType aSlice = m_aData[ ::std::slice( nColumnIndex, m_nRowCount, m_nColumnCount ) ];
183*b1cdbd2cSJim Jagielski     for( vector< double >::size_type i = 0; i < rNewData.size(); ++i )
184*b1cdbd2cSJim Jagielski         aSlice[i] = rNewData[i];
185*b1cdbd2cSJim Jagielski     m_aData[ ::std::slice( nColumnIndex, m_nRowCount, m_nColumnCount ) ] = aSlice;
186*b1cdbd2cSJim Jagielski }
187*b1cdbd2cSJim Jagielski 
setRowValues(sal_Int32 nRowIndex,const vector<double> & rNewData)188*b1cdbd2cSJim Jagielski void InternalData::setRowValues( sal_Int32 nRowIndex, const vector< double > & rNewData )
189*b1cdbd2cSJim Jagielski {
190*b1cdbd2cSJim Jagielski     if( nRowIndex < 0 )
191*b1cdbd2cSJim Jagielski         return;
192*b1cdbd2cSJim Jagielski     enlargeData( rNewData.size(), nRowIndex+1 );
193*b1cdbd2cSJim Jagielski 
194*b1cdbd2cSJim Jagielski     tDataType aSlice = m_aData[ ::std::slice( nRowIndex*m_nColumnCount, m_nColumnCount, 1 ) ];
195*b1cdbd2cSJim Jagielski     for( vector< double >::size_type i = 0; i < rNewData.size(); ++i )
196*b1cdbd2cSJim Jagielski         aSlice[i] = rNewData[i];
197*b1cdbd2cSJim Jagielski     m_aData[ ::std::slice( nRowIndex*m_nColumnCount, m_nColumnCount, 1 ) ]= aSlice;
198*b1cdbd2cSJim Jagielski }
199*b1cdbd2cSJim Jagielski 
setComplexColumnLabel(sal_Int32 nColumnIndex,const vector<uno::Any> & rComplexLabel)200*b1cdbd2cSJim Jagielski void InternalData::setComplexColumnLabel( sal_Int32 nColumnIndex, const vector< uno::Any >& rComplexLabel )
201*b1cdbd2cSJim Jagielski {
202*b1cdbd2cSJim Jagielski     if( nColumnIndex < 0 )
203*b1cdbd2cSJim Jagielski         return;
204*b1cdbd2cSJim Jagielski     if( nColumnIndex >= static_cast< sal_Int32 >( m_aColumnLabels.size() ) )
205*b1cdbd2cSJim Jagielski     {
206*b1cdbd2cSJim Jagielski         m_aColumnLabels.resize(nColumnIndex+1);
207*b1cdbd2cSJim Jagielski         enlargeData( nColumnIndex+1, 0 );
208*b1cdbd2cSJim Jagielski     }
209*b1cdbd2cSJim Jagielski     m_aColumnLabels[nColumnIndex]=rComplexLabel;
210*b1cdbd2cSJim Jagielski }
211*b1cdbd2cSJim Jagielski 
setComplexRowLabel(sal_Int32 nRowIndex,const vector<uno::Any> & rComplexLabel)212*b1cdbd2cSJim Jagielski void InternalData::setComplexRowLabel( sal_Int32 nRowIndex, const vector< uno::Any >& rComplexLabel )
213*b1cdbd2cSJim Jagielski {
214*b1cdbd2cSJim Jagielski     if( nRowIndex < 0 )
215*b1cdbd2cSJim Jagielski         return;
216*b1cdbd2cSJim Jagielski     if( nRowIndex >= static_cast< sal_Int32 >( m_aRowLabels.size() ) )
217*b1cdbd2cSJim Jagielski     {
218*b1cdbd2cSJim Jagielski         m_aRowLabels.resize(nRowIndex+1);
219*b1cdbd2cSJim Jagielski         enlargeData( 0, nRowIndex+1 );
220*b1cdbd2cSJim Jagielski     }
221*b1cdbd2cSJim Jagielski     m_aRowLabels[nRowIndex] = rComplexLabel;
222*b1cdbd2cSJim Jagielski }
223*b1cdbd2cSJim Jagielski 
getComplexColumnLabel(sal_Int32 nColumnIndex) const224*b1cdbd2cSJim Jagielski vector< uno::Any > InternalData::getComplexColumnLabel( sal_Int32 nColumnIndex ) const
225*b1cdbd2cSJim Jagielski {
226*b1cdbd2cSJim Jagielski     if( nColumnIndex < static_cast< sal_Int32 >( m_aColumnLabels.size() ) )
227*b1cdbd2cSJim Jagielski         return m_aColumnLabels[nColumnIndex];
228*b1cdbd2cSJim Jagielski     else
229*b1cdbd2cSJim Jagielski         return vector< uno::Any >();
230*b1cdbd2cSJim Jagielski }
getComplexRowLabel(sal_Int32 nRowIndex) const231*b1cdbd2cSJim Jagielski vector< uno::Any > InternalData::getComplexRowLabel( sal_Int32 nRowIndex ) const
232*b1cdbd2cSJim Jagielski {
233*b1cdbd2cSJim Jagielski     if( nRowIndex < static_cast< sal_Int32 >( m_aRowLabels.size() ) )
234*b1cdbd2cSJim Jagielski         return m_aRowLabels[nRowIndex];
235*b1cdbd2cSJim Jagielski     else
236*b1cdbd2cSJim Jagielski         return vector< uno::Any >();
237*b1cdbd2cSJim Jagielski }
238*b1cdbd2cSJim Jagielski 
swapRowWithNext(sal_Int32 nRowIndex)239*b1cdbd2cSJim Jagielski void InternalData::swapRowWithNext( sal_Int32 nRowIndex )
240*b1cdbd2cSJim Jagielski {
241*b1cdbd2cSJim Jagielski     if( nRowIndex < m_nRowCount - 1 )
242*b1cdbd2cSJim Jagielski     {
243*b1cdbd2cSJim Jagielski         const sal_Int32 nMax = m_nColumnCount;
244*b1cdbd2cSJim Jagielski         for( sal_Int32 nColIdx=0; nColIdx<nMax; ++nColIdx )
245*b1cdbd2cSJim Jagielski         {
246*b1cdbd2cSJim Jagielski             size_t nIndex1 = nColIdx + nRowIndex*m_nColumnCount;
247*b1cdbd2cSJim Jagielski             size_t nIndex2 = nIndex1 + m_nColumnCount;
248*b1cdbd2cSJim Jagielski             double fTemp = m_aData[nIndex1];
249*b1cdbd2cSJim Jagielski             m_aData[nIndex1] = m_aData[nIndex2];
250*b1cdbd2cSJim Jagielski             m_aData[nIndex2] = fTemp;
251*b1cdbd2cSJim Jagielski         }
252*b1cdbd2cSJim Jagielski 
253*b1cdbd2cSJim Jagielski         vector< uno::Any > aTemp( m_aRowLabels[nRowIndex] );
254*b1cdbd2cSJim Jagielski         m_aRowLabels[nRowIndex] = m_aRowLabels[nRowIndex + 1];
255*b1cdbd2cSJim Jagielski         m_aRowLabels[nRowIndex + 1] = aTemp;
256*b1cdbd2cSJim Jagielski     }
257*b1cdbd2cSJim Jagielski }
258*b1cdbd2cSJim Jagielski 
swapColumnWithNext(sal_Int32 nColumnIndex)259*b1cdbd2cSJim Jagielski void InternalData::swapColumnWithNext( sal_Int32 nColumnIndex )
260*b1cdbd2cSJim Jagielski {
261*b1cdbd2cSJim Jagielski     if( nColumnIndex < m_nColumnCount - 1 )
262*b1cdbd2cSJim Jagielski     {
263*b1cdbd2cSJim Jagielski         const sal_Int32 nMax = m_nRowCount;
264*b1cdbd2cSJim Jagielski         for( sal_Int32 nRowIdx=0; nRowIdx<nMax; ++nRowIdx )
265*b1cdbd2cSJim Jagielski         {
266*b1cdbd2cSJim Jagielski             size_t nIndex1 = nColumnIndex + nRowIdx*m_nColumnCount;
267*b1cdbd2cSJim Jagielski             size_t nIndex2 = nIndex1 + 1;
268*b1cdbd2cSJim Jagielski             double fTemp = m_aData[nIndex1];
269*b1cdbd2cSJim Jagielski             m_aData[nIndex1] = m_aData[nIndex2];
270*b1cdbd2cSJim Jagielski             m_aData[nIndex2] = fTemp;
271*b1cdbd2cSJim Jagielski         }
272*b1cdbd2cSJim Jagielski 
273*b1cdbd2cSJim Jagielski         vector< uno::Any > aTemp( m_aColumnLabels[nColumnIndex] );
274*b1cdbd2cSJim Jagielski         m_aColumnLabels[nColumnIndex] = m_aColumnLabels[nColumnIndex + 1];
275*b1cdbd2cSJim Jagielski         m_aColumnLabels[nColumnIndex + 1] = aTemp;
276*b1cdbd2cSJim Jagielski     }
277*b1cdbd2cSJim Jagielski }
278*b1cdbd2cSJim Jagielski 
enlargeData(sal_Int32 nColumnCount,sal_Int32 nRowCount)279*b1cdbd2cSJim Jagielski bool InternalData::enlargeData( sal_Int32 nColumnCount, sal_Int32 nRowCount )
280*b1cdbd2cSJim Jagielski {
281*b1cdbd2cSJim Jagielski     sal_Int32 nNewColumnCount( ::std::max<sal_Int32>( m_nColumnCount, nColumnCount ) );
282*b1cdbd2cSJim Jagielski     sal_Int32 nNewRowCount( ::std::max<sal_Int32>( m_nRowCount, nRowCount ) );
283*b1cdbd2cSJim Jagielski     sal_Int32 nNewSize( nNewColumnCount*nNewRowCount );
284*b1cdbd2cSJim Jagielski 
285*b1cdbd2cSJim Jagielski     bool bGrow = (nNewSize > m_nColumnCount*m_nRowCount);
286*b1cdbd2cSJim Jagielski 
287*b1cdbd2cSJim Jagielski     if( bGrow )
288*b1cdbd2cSJim Jagielski     {
289*b1cdbd2cSJim Jagielski         double fNan;
290*b1cdbd2cSJim Jagielski         ::rtl::math::setNan( &fNan );
291*b1cdbd2cSJim Jagielski         tDataType aNewData( fNan, nNewSize );
292*b1cdbd2cSJim Jagielski         // copy old data
293*b1cdbd2cSJim Jagielski         for( int nCol=0; nCol<m_nColumnCount; ++nCol )
294*b1cdbd2cSJim Jagielski             static_cast< tDataType >(
295*b1cdbd2cSJim Jagielski                 aNewData[ ::std::slice( nCol, m_nRowCount, nNewColumnCount ) ] ) =
296*b1cdbd2cSJim Jagielski                 m_aData[ ::std::slice( nCol, m_nRowCount, m_nColumnCount ) ];
297*b1cdbd2cSJim Jagielski 
298*b1cdbd2cSJim Jagielski         m_aData.resize( nNewSize );
299*b1cdbd2cSJim Jagielski         m_aData = aNewData;
300*b1cdbd2cSJim Jagielski     }
301*b1cdbd2cSJim Jagielski     m_nColumnCount = nNewColumnCount;
302*b1cdbd2cSJim Jagielski     m_nRowCount = nNewRowCount;
303*b1cdbd2cSJim Jagielski     return bGrow;
304*b1cdbd2cSJim Jagielski }
305*b1cdbd2cSJim Jagielski 
insertColumn(sal_Int32 nAfterIndex)306*b1cdbd2cSJim Jagielski void InternalData::insertColumn( sal_Int32 nAfterIndex )
307*b1cdbd2cSJim Jagielski {
308*b1cdbd2cSJim Jagielski     // note: -1 is allowed, as we insert after the given index
309*b1cdbd2cSJim Jagielski     OSL_ASSERT( nAfterIndex < m_nColumnCount && nAfterIndex >= -1 );
310*b1cdbd2cSJim Jagielski     if( nAfterIndex >= m_nColumnCount || nAfterIndex < -1 )
311*b1cdbd2cSJim Jagielski         return;
312*b1cdbd2cSJim Jagielski     sal_Int32 nNewColumnCount = m_nColumnCount + 1;
313*b1cdbd2cSJim Jagielski     sal_Int32 nNewSize( nNewColumnCount * m_nRowCount );
314*b1cdbd2cSJim Jagielski 
315*b1cdbd2cSJim Jagielski     double fNan;
316*b1cdbd2cSJim Jagielski     ::rtl::math::setNan( &fNan );
317*b1cdbd2cSJim Jagielski     tDataType aNewData( fNan, nNewSize );
318*b1cdbd2cSJim Jagielski 
319*b1cdbd2cSJim Jagielski     // copy old data
320*b1cdbd2cSJim Jagielski     int nCol=0;
321*b1cdbd2cSJim Jagielski     for( ; nCol<=nAfterIndex; ++nCol )
322*b1cdbd2cSJim Jagielski         aNewData[ ::std::slice( nCol, m_nRowCount, nNewColumnCount ) ] =
323*b1cdbd2cSJim Jagielski             static_cast< tDataType >(
324*b1cdbd2cSJim Jagielski                 m_aData[ ::std::slice( nCol, m_nRowCount, m_nColumnCount ) ] );
325*b1cdbd2cSJim Jagielski     for( ++nCol; nCol<nNewColumnCount; ++nCol )
326*b1cdbd2cSJim Jagielski         aNewData[ ::std::slice( nCol, m_nRowCount, nNewColumnCount ) ] =
327*b1cdbd2cSJim Jagielski             static_cast< tDataType >(
328*b1cdbd2cSJim Jagielski                 m_aData[ ::std::slice( nCol - 1, m_nRowCount, m_nColumnCount ) ] );
329*b1cdbd2cSJim Jagielski 
330*b1cdbd2cSJim Jagielski     m_nColumnCount = nNewColumnCount;
331*b1cdbd2cSJim Jagielski     m_aData.resize( nNewSize );
332*b1cdbd2cSJim Jagielski     m_aData = aNewData;
333*b1cdbd2cSJim Jagielski 
334*b1cdbd2cSJim Jagielski     // labels
335*b1cdbd2cSJim Jagielski     if( nAfterIndex < static_cast< sal_Int32 >( m_aColumnLabels.size()))
336*b1cdbd2cSJim Jagielski         m_aColumnLabels.insert( m_aColumnLabels.begin() + (nAfterIndex + 1), vector< uno::Any >(1) );
337*b1cdbd2cSJim Jagielski 
338*b1cdbd2cSJim Jagielski #if OSL_DEBUG_LEVEL > 2
339*b1cdbd2cSJim Jagielski     traceData();
340*b1cdbd2cSJim Jagielski #endif
341*b1cdbd2cSJim Jagielski }
342*b1cdbd2cSJim Jagielski 
appendColumn()343*b1cdbd2cSJim Jagielski sal_Int32 InternalData::appendColumn()
344*b1cdbd2cSJim Jagielski {
345*b1cdbd2cSJim Jagielski     insertColumn( getColumnCount() - 1 );
346*b1cdbd2cSJim Jagielski     return getColumnCount() - 1;
347*b1cdbd2cSJim Jagielski }
348*b1cdbd2cSJim Jagielski 
appendRow()349*b1cdbd2cSJim Jagielski sal_Int32 InternalData::appendRow()
350*b1cdbd2cSJim Jagielski {
351*b1cdbd2cSJim Jagielski     insertRow( getRowCount() - 1 );
352*b1cdbd2cSJim Jagielski     return getRowCount() - 1;
353*b1cdbd2cSJim Jagielski }
354*b1cdbd2cSJim Jagielski 
insertRow(sal_Int32 nAfterIndex)355*b1cdbd2cSJim Jagielski void InternalData::insertRow( sal_Int32 nAfterIndex )
356*b1cdbd2cSJim Jagielski {
357*b1cdbd2cSJim Jagielski     // note: -1 is allowed, as we insert after the given index
358*b1cdbd2cSJim Jagielski     OSL_ASSERT( nAfterIndex < m_nRowCount && nAfterIndex >= -1 );
359*b1cdbd2cSJim Jagielski     if( nAfterIndex >= m_nRowCount || nAfterIndex < -1 )
360*b1cdbd2cSJim Jagielski         return;
361*b1cdbd2cSJim Jagielski     sal_Int32 nNewRowCount = m_nRowCount + 1;
362*b1cdbd2cSJim Jagielski     sal_Int32 nNewSize( m_nColumnCount * nNewRowCount );
363*b1cdbd2cSJim Jagielski 
364*b1cdbd2cSJim Jagielski     double fNan;
365*b1cdbd2cSJim Jagielski     ::rtl::math::setNan( &fNan );
366*b1cdbd2cSJim Jagielski     tDataType aNewData( fNan, nNewSize );
367*b1cdbd2cSJim Jagielski 
368*b1cdbd2cSJim Jagielski     // copy old data
369*b1cdbd2cSJim Jagielski     sal_Int32 nIndex = nAfterIndex + 1;
370*b1cdbd2cSJim Jagielski     aNewData[ ::std::slice( 0, nIndex * m_nColumnCount, 1 ) ] =
371*b1cdbd2cSJim Jagielski         static_cast< tDataType >(
372*b1cdbd2cSJim Jagielski             m_aData[ ::std::slice( 0, nIndex * m_nColumnCount, 1 ) ] );
373*b1cdbd2cSJim Jagielski 
374*b1cdbd2cSJim Jagielski     if( nIndex < m_nRowCount )
375*b1cdbd2cSJim Jagielski     {
376*b1cdbd2cSJim Jagielski         sal_Int32 nRemainingCount = m_nColumnCount * (m_nRowCount - nIndex);
377*b1cdbd2cSJim Jagielski         aNewData[ ::std::slice( (nIndex + 1) * m_nColumnCount, nRemainingCount, 1 ) ] =
378*b1cdbd2cSJim Jagielski             static_cast< tDataType >(
379*b1cdbd2cSJim Jagielski                 m_aData[ ::std::slice( nIndex * m_nColumnCount, nRemainingCount, 1 ) ] );
380*b1cdbd2cSJim Jagielski     }
381*b1cdbd2cSJim Jagielski 
382*b1cdbd2cSJim Jagielski     m_nRowCount = nNewRowCount;
383*b1cdbd2cSJim Jagielski     m_aData.resize( nNewSize );
384*b1cdbd2cSJim Jagielski     m_aData = aNewData;
385*b1cdbd2cSJim Jagielski 
386*b1cdbd2cSJim Jagielski     // labels
387*b1cdbd2cSJim Jagielski     if( nAfterIndex < static_cast< sal_Int32 >( m_aRowLabels.size()))
388*b1cdbd2cSJim Jagielski         m_aRowLabels.insert( m_aRowLabels.begin() + nIndex, vector< uno::Any > (1));
389*b1cdbd2cSJim Jagielski 
390*b1cdbd2cSJim Jagielski #if OSL_DEBUG_LEVEL > 2
391*b1cdbd2cSJim Jagielski     traceData();
392*b1cdbd2cSJim Jagielski #endif
393*b1cdbd2cSJim Jagielski }
394*b1cdbd2cSJim Jagielski 
deleteColumn(sal_Int32 nAtIndex)395*b1cdbd2cSJim Jagielski void InternalData::deleteColumn( sal_Int32 nAtIndex )
396*b1cdbd2cSJim Jagielski {
397*b1cdbd2cSJim Jagielski     OSL_ASSERT( nAtIndex < m_nColumnCount && nAtIndex >= 0 );
398*b1cdbd2cSJim Jagielski     if( nAtIndex >= m_nColumnCount || m_nColumnCount < 1 || nAtIndex < 0 )
399*b1cdbd2cSJim Jagielski         return;
400*b1cdbd2cSJim Jagielski     sal_Int32 nNewColumnCount = m_nColumnCount - 1;
401*b1cdbd2cSJim Jagielski     sal_Int32 nNewSize( nNewColumnCount * m_nRowCount );
402*b1cdbd2cSJim Jagielski 
403*b1cdbd2cSJim Jagielski     double fNan;
404*b1cdbd2cSJim Jagielski     ::rtl::math::setNan( &fNan );
405*b1cdbd2cSJim Jagielski     tDataType aNewData( fNan, nNewSize );
406*b1cdbd2cSJim Jagielski 
407*b1cdbd2cSJim Jagielski     // copy old data
408*b1cdbd2cSJim Jagielski     int nCol=0;
409*b1cdbd2cSJim Jagielski     for( ; nCol<nAtIndex; ++nCol )
410*b1cdbd2cSJim Jagielski         aNewData[ ::std::slice( nCol, m_nRowCount, nNewColumnCount ) ] =
411*b1cdbd2cSJim Jagielski             static_cast< tDataType >(
412*b1cdbd2cSJim Jagielski                 m_aData[ ::std::slice( nCol, m_nRowCount, m_nColumnCount ) ] );
413*b1cdbd2cSJim Jagielski     for( ; nCol<nNewColumnCount; ++nCol )
414*b1cdbd2cSJim Jagielski         aNewData[ ::std::slice( nCol, m_nRowCount, nNewColumnCount ) ] =
415*b1cdbd2cSJim Jagielski             static_cast< tDataType >(
416*b1cdbd2cSJim Jagielski                 m_aData[ ::std::slice( nCol + 1, m_nRowCount, m_nColumnCount ) ] );
417*b1cdbd2cSJim Jagielski 
418*b1cdbd2cSJim Jagielski     m_nColumnCount = nNewColumnCount;
419*b1cdbd2cSJim Jagielski     m_aData.resize( nNewSize );
420*b1cdbd2cSJim Jagielski     m_aData = aNewData;
421*b1cdbd2cSJim Jagielski 
422*b1cdbd2cSJim Jagielski     // labels
423*b1cdbd2cSJim Jagielski     if( nAtIndex < static_cast< sal_Int32 >( m_aColumnLabels.size()))
424*b1cdbd2cSJim Jagielski         m_aColumnLabels.erase( m_aColumnLabels.begin() + nAtIndex );
425*b1cdbd2cSJim Jagielski 
426*b1cdbd2cSJim Jagielski #if OSL_DEBUG_LEVEL > 2
427*b1cdbd2cSJim Jagielski     traceData();
428*b1cdbd2cSJim Jagielski #endif
429*b1cdbd2cSJim Jagielski }
430*b1cdbd2cSJim Jagielski 
deleteRow(sal_Int32 nAtIndex)431*b1cdbd2cSJim Jagielski void InternalData::deleteRow( sal_Int32 nAtIndex )
432*b1cdbd2cSJim Jagielski {
433*b1cdbd2cSJim Jagielski     OSL_ASSERT( nAtIndex < m_nRowCount && nAtIndex >= 0 );
434*b1cdbd2cSJim Jagielski     if( nAtIndex >= m_nRowCount || m_nRowCount < 1 || nAtIndex < 0 )
435*b1cdbd2cSJim Jagielski         return;
436*b1cdbd2cSJim Jagielski     sal_Int32 nNewRowCount = m_nRowCount - 1;
437*b1cdbd2cSJim Jagielski     sal_Int32 nNewSize( m_nColumnCount * nNewRowCount );
438*b1cdbd2cSJim Jagielski 
439*b1cdbd2cSJim Jagielski     double fNan;
440*b1cdbd2cSJim Jagielski     ::rtl::math::setNan( &fNan );
441*b1cdbd2cSJim Jagielski     tDataType aNewData( fNan, nNewSize );
442*b1cdbd2cSJim Jagielski 
443*b1cdbd2cSJim Jagielski     // copy old data
444*b1cdbd2cSJim Jagielski     sal_Int32 nIndex = nAtIndex;
445*b1cdbd2cSJim Jagielski     if( nIndex )
446*b1cdbd2cSJim Jagielski         aNewData[ ::std::slice( 0, nIndex * m_nColumnCount, 1 ) ] =
447*b1cdbd2cSJim Jagielski             static_cast< tDataType >(
448*b1cdbd2cSJim Jagielski                 m_aData[ ::std::slice( 0, nIndex * m_nColumnCount, 1 ) ] );
449*b1cdbd2cSJim Jagielski 
450*b1cdbd2cSJim Jagielski     if( nIndex < nNewRowCount )
451*b1cdbd2cSJim Jagielski     {
452*b1cdbd2cSJim Jagielski         sal_Int32 nRemainingCount = m_nColumnCount * (nNewRowCount - nIndex);
453*b1cdbd2cSJim Jagielski         aNewData[ ::std::slice( nIndex * m_nColumnCount, nRemainingCount, 1 ) ] =
454*b1cdbd2cSJim Jagielski             static_cast< tDataType >(
455*b1cdbd2cSJim Jagielski                 m_aData[ ::std::slice( (nIndex + 1) * m_nColumnCount, nRemainingCount, 1 ) ] );
456*b1cdbd2cSJim Jagielski     }
457*b1cdbd2cSJim Jagielski 
458*b1cdbd2cSJim Jagielski     m_nRowCount = nNewRowCount;
459*b1cdbd2cSJim Jagielski     m_aData.resize( nNewSize );
460*b1cdbd2cSJim Jagielski     m_aData = aNewData;
461*b1cdbd2cSJim Jagielski 
462*b1cdbd2cSJim Jagielski     // labels
463*b1cdbd2cSJim Jagielski     if( nAtIndex < static_cast< sal_Int32 >( m_aRowLabels.size()))
464*b1cdbd2cSJim Jagielski         m_aRowLabels.erase( m_aRowLabels.begin() + nAtIndex );
465*b1cdbd2cSJim Jagielski 
466*b1cdbd2cSJim Jagielski #if OSL_DEBUG_LEVEL > 2
467*b1cdbd2cSJim Jagielski     traceData();
468*b1cdbd2cSJim Jagielski #endif
469*b1cdbd2cSJim Jagielski }
470*b1cdbd2cSJim Jagielski 
getRowCount() const471*b1cdbd2cSJim Jagielski sal_Int32 InternalData::getRowCount() const
472*b1cdbd2cSJim Jagielski {
473*b1cdbd2cSJim Jagielski     return m_nRowCount;
474*b1cdbd2cSJim Jagielski }
475*b1cdbd2cSJim Jagielski 
getColumnCount() const476*b1cdbd2cSJim Jagielski sal_Int32 InternalData::getColumnCount() const
477*b1cdbd2cSJim Jagielski {
478*b1cdbd2cSJim Jagielski     return m_nColumnCount;
479*b1cdbd2cSJim Jagielski }
480*b1cdbd2cSJim Jagielski 
setComplexRowLabels(const vector<vector<uno::Any>> & rNewRowLabels)481*b1cdbd2cSJim Jagielski void InternalData::setComplexRowLabels( const vector< vector< uno::Any > >& rNewRowLabels )
482*b1cdbd2cSJim Jagielski {
483*b1cdbd2cSJim Jagielski     m_aRowLabels = rNewRowLabels;
484*b1cdbd2cSJim Jagielski     sal_Int32 nNewRowCount = static_cast< sal_Int32 >( m_aRowLabels.size() );
485*b1cdbd2cSJim Jagielski     if( nNewRowCount < m_nRowCount )
486*b1cdbd2cSJim Jagielski         m_aRowLabels.resize( m_nRowCount );
487*b1cdbd2cSJim Jagielski     else
488*b1cdbd2cSJim Jagielski         enlargeData( 0, nNewRowCount );
489*b1cdbd2cSJim Jagielski }
490*b1cdbd2cSJim Jagielski 
getComplexRowLabels() const491*b1cdbd2cSJim Jagielski vector< vector< uno::Any > > InternalData::getComplexRowLabels() const
492*b1cdbd2cSJim Jagielski {
493*b1cdbd2cSJim Jagielski     return m_aRowLabels;
494*b1cdbd2cSJim Jagielski }
495*b1cdbd2cSJim Jagielski 
setComplexColumnLabels(const vector<vector<uno::Any>> & rNewColumnLabels)496*b1cdbd2cSJim Jagielski void InternalData::setComplexColumnLabels( const vector< vector< uno::Any > >& rNewColumnLabels )
497*b1cdbd2cSJim Jagielski {
498*b1cdbd2cSJim Jagielski     m_aColumnLabels = rNewColumnLabels;
499*b1cdbd2cSJim Jagielski     sal_Int32 nNewColumnCount = static_cast< sal_Int32 >( m_aColumnLabels.size() );
500*b1cdbd2cSJim Jagielski     if( nNewColumnCount < m_nColumnCount )
501*b1cdbd2cSJim Jagielski         m_aColumnLabels.resize( m_nColumnCount );
502*b1cdbd2cSJim Jagielski     else
503*b1cdbd2cSJim Jagielski         enlargeData( nNewColumnCount, 0 );
504*b1cdbd2cSJim Jagielski }
505*b1cdbd2cSJim Jagielski 
getComplexColumnLabels() const506*b1cdbd2cSJim Jagielski vector< vector< uno::Any > > InternalData::getComplexColumnLabels() const
507*b1cdbd2cSJim Jagielski {
508*b1cdbd2cSJim Jagielski     return m_aColumnLabels;
509*b1cdbd2cSJim Jagielski }
510*b1cdbd2cSJim Jagielski 
511*b1cdbd2cSJim Jagielski #if OSL_DEBUG_LEVEL > 2
traceData() const512*b1cdbd2cSJim Jagielski void InternalData::traceData() const
513*b1cdbd2cSJim Jagielski {
514*b1cdbd2cSJim Jagielski     OSL_TRACE( "InternalData: Data in rows\n" );
515*b1cdbd2cSJim Jagielski 
516*b1cdbd2cSJim Jagielski     for( sal_Int32 i=0; i<m_nRowCount; ++i )
517*b1cdbd2cSJim Jagielski     {
518*b1cdbd2cSJim Jagielski         tDataType aSlice( m_aData[ ::std::slice( i*m_nColumnCount, m_nColumnCount, 1 ) ] );
519*b1cdbd2cSJim Jagielski         for( sal_Int32 j=0; j<m_nColumnCount; ++j )
520*b1cdbd2cSJim Jagielski             OSL_TRACE( "%lf ", aSlice[j] );
521*b1cdbd2cSJim Jagielski         OSL_TRACE( "\n" );
522*b1cdbd2cSJim Jagielski     }
523*b1cdbd2cSJim Jagielski     OSL_TRACE( "\n" );
524*b1cdbd2cSJim Jagielski }
525*b1cdbd2cSJim Jagielski #endif
526*b1cdbd2cSJim Jagielski 
527*b1cdbd2cSJim Jagielski } //  namespace chart
528