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_extensions.hxx"
26*b1cdbd2cSJim Jagielski #include <grid.hrc>
27*b1cdbd2cSJim Jagielski #include <cstdio>
28*b1cdbd2cSJim Jagielski #include <math.h> // for M_LN10 and M_E
29*b1cdbd2cSJim Jagielski 
30*b1cdbd2cSJim Jagielski #define _USE_MATH_DEFINES
31*b1cdbd2cSJim Jagielski #include <cmath>
32*b1cdbd2cSJim Jagielski #undef _USE_MATH_DEFINES
33*b1cdbd2cSJim Jagielski 
34*b1cdbd2cSJim Jagielski #include <grid.hxx>
35*b1cdbd2cSJim Jagielski 
36*b1cdbd2cSJim Jagielski // for ::std::sort
37*b1cdbd2cSJim Jagielski #include <algorithm>
38*b1cdbd2cSJim Jagielski 
39*b1cdbd2cSJim Jagielski ResId SaneResId( sal_uInt32 );
40*b1cdbd2cSJim Jagielski 
41*b1cdbd2cSJim Jagielski /***********************************************************************
42*b1cdbd2cSJim Jagielski  *
43*b1cdbd2cSJim Jagielski  *	GridWindow
44*b1cdbd2cSJim Jagielski  *
45*b1cdbd2cSJim Jagielski  ***********************************************************************/
46*b1cdbd2cSJim Jagielski 
47*b1cdbd2cSJim Jagielski // ---------------------------------------------------------------------
48*b1cdbd2cSJim Jagielski 
GridWindow(double * pXValues,double * pYValues,int nValues,Window * pParent,sal_Bool bCutValues)49*b1cdbd2cSJim Jagielski GridWindow::GridWindow(double* pXValues, double* pYValues, int nValues, Window* pParent, sal_Bool bCutValues )
50*b1cdbd2cSJim Jagielski :	ModalDialog( pParent, SaneResId( GRID_DIALOG ) ),
51*b1cdbd2cSJim Jagielski 	m_aGridArea( 50, 15, 100, 100 ),
52*b1cdbd2cSJim Jagielski 	m_pXValues( pXValues ),
53*b1cdbd2cSJim Jagielski 	m_pOrigYValues( pYValues ),
54*b1cdbd2cSJim Jagielski 	m_nValues( nValues ),
55*b1cdbd2cSJim Jagielski 	m_pNewYValues( NULL ),
56*b1cdbd2cSJim Jagielski 	m_bCutValues( bCutValues ),
57*b1cdbd2cSJim Jagielski 	m_aHandles(),
58*b1cdbd2cSJim Jagielski 	m_nDragIndex( 0xffffffff ),
59*b1cdbd2cSJim Jagielski 	m_aMarkerBitmap( Bitmap( SaneResId( GRID_DIALOG_HANDLE_BMP ) ), Color( 255, 255, 255 ) ),
60*b1cdbd2cSJim Jagielski 	m_aOKButton( this, SaneResId( GRID_DIALOG_OK_BTN ) ),
61*b1cdbd2cSJim Jagielski 	m_aCancelButton( this, SaneResId( GRID_DIALOG_CANCEL_BTN ) ),
62*b1cdbd2cSJim Jagielski 	m_aResetTypeBox( this, SaneResId( GRID_DIALOG_TYPE_BOX ) ),
63*b1cdbd2cSJim Jagielski 	m_aResetButton( this, SaneResId( GRID_DIALOG_RESET_BTN ) )
64*b1cdbd2cSJim Jagielski {
65*b1cdbd2cSJim Jagielski 	sal_uInt16 nPos = m_aResetTypeBox.InsertEntry( String( SaneResId( RESET_TYPE_LINEAR_ASCENDING ) ) );
66*b1cdbd2cSJim Jagielski 	m_aResetTypeBox.SetEntryData( nPos, (void *)RESET_TYPE_LINEAR_ASCENDING );
67*b1cdbd2cSJim Jagielski 
68*b1cdbd2cSJim Jagielski 	nPos = m_aResetTypeBox.InsertEntry( String( SaneResId( RESET_TYPE_LINEAR_DESCENDING ) ) );
69*b1cdbd2cSJim Jagielski 	m_aResetTypeBox.SetEntryData( nPos, (void *)RESET_TYPE_LINEAR_DESCENDING );
70*b1cdbd2cSJim Jagielski 
71*b1cdbd2cSJim Jagielski 	nPos = m_aResetTypeBox.InsertEntry( String( SaneResId( RESET_TYPE_RESET ) ) );
72*b1cdbd2cSJim Jagielski 	m_aResetTypeBox.SetEntryData( nPos, (void *)RESET_TYPE_RESET );
73*b1cdbd2cSJim Jagielski 
74*b1cdbd2cSJim Jagielski 	nPos = m_aResetTypeBox.InsertEntry( String( SaneResId( RESET_TYPE_EXPONENTIAL ) ) );
75*b1cdbd2cSJim Jagielski 	m_aResetTypeBox.SetEntryData( nPos, (void *)RESET_TYPE_EXPONENTIAL );
76*b1cdbd2cSJim Jagielski 
77*b1cdbd2cSJim Jagielski 	m_aResetTypeBox.SelectEntryPos( 0 );
78*b1cdbd2cSJim Jagielski 
79*b1cdbd2cSJim Jagielski 	m_aResetButton.SetClickHdl( LINK( this, GridWindow, ClickButtonHdl ) );
80*b1cdbd2cSJim Jagielski 
81*b1cdbd2cSJim Jagielski 	SetMapMode( MapMode( MAP_PIXEL ) );
82*b1cdbd2cSJim Jagielski 	Size aSize = GetOutputSizePixel();
83*b1cdbd2cSJim Jagielski 	Size aBtnSize = m_aOKButton.GetOutputSizePixel();
84*b1cdbd2cSJim Jagielski 	m_aGridArea.setWidth( aSize.Width() - aBtnSize.Width() - 80 );
85*b1cdbd2cSJim Jagielski 	m_aGridArea.setHeight( aSize.Height() - 40 );
86*b1cdbd2cSJim Jagielski 
87*b1cdbd2cSJim Jagielski 	if( m_pOrigYValues && m_nValues )
88*b1cdbd2cSJim Jagielski 	{
89*b1cdbd2cSJim Jagielski 		m_pNewYValues = new double[ m_nValues ];
90*b1cdbd2cSJim Jagielski 		memcpy( m_pNewYValues, m_pOrigYValues, sizeof( double ) * m_nValues );
91*b1cdbd2cSJim Jagielski 	}
92*b1cdbd2cSJim Jagielski 
93*b1cdbd2cSJim Jagielski 	setBoundings( 0, 0, 1023, 1023 );
94*b1cdbd2cSJim Jagielski 	computeExtremes();
95*b1cdbd2cSJim Jagielski 
96*b1cdbd2cSJim Jagielski 	// create left and right marker as first and last entry
97*b1cdbd2cSJim Jagielski 	m_BmOffX = sal_uInt16(m_aMarkerBitmap.GetSizePixel().Width() >> 1);
98*b1cdbd2cSJim Jagielski 	m_BmOffY = sal_uInt16(m_aMarkerBitmap.GetSizePixel().Height() >> 1);
99*b1cdbd2cSJim Jagielski 	m_aHandles.push_back(impHandle(transform(findMinX(), findMinY()), m_BmOffX, m_BmOffY));
100*b1cdbd2cSJim Jagielski 	m_aHandles.push_back(impHandle(transform(findMaxX(), findMaxY()), m_BmOffX, m_BmOffY));
101*b1cdbd2cSJim Jagielski 
102*b1cdbd2cSJim Jagielski 	FreeResource();
103*b1cdbd2cSJim Jagielski }
104*b1cdbd2cSJim Jagielski 
105*b1cdbd2cSJim Jagielski // ---------------------------------------------------------------------
106*b1cdbd2cSJim Jagielski 
~GridWindow()107*b1cdbd2cSJim Jagielski GridWindow::~GridWindow()
108*b1cdbd2cSJim Jagielski {
109*b1cdbd2cSJim Jagielski 	if( m_pNewYValues )
110*b1cdbd2cSJim Jagielski 		delete [] m_pNewYValues;
111*b1cdbd2cSJim Jagielski }
112*b1cdbd2cSJim Jagielski 
113*b1cdbd2cSJim Jagielski // ---------------------------------------------------------------------
114*b1cdbd2cSJim Jagielski 
findMinX()115*b1cdbd2cSJim Jagielski double GridWindow::findMinX()
116*b1cdbd2cSJim Jagielski {
117*b1cdbd2cSJim Jagielski 	if( ! m_pXValues )
118*b1cdbd2cSJim Jagielski 		return 0.0;
119*b1cdbd2cSJim Jagielski 	double fMin = m_pXValues[0];
120*b1cdbd2cSJim Jagielski 	for( int i = 1; i < m_nValues; i++ )
121*b1cdbd2cSJim Jagielski 		if( m_pXValues[ i ] < fMin )
122*b1cdbd2cSJim Jagielski 			fMin = m_pXValues[ i ];
123*b1cdbd2cSJim Jagielski 	return fMin;
124*b1cdbd2cSJim Jagielski }
125*b1cdbd2cSJim Jagielski 
126*b1cdbd2cSJim Jagielski // ---------------------------------------------------------------------
127*b1cdbd2cSJim Jagielski 
findMinY()128*b1cdbd2cSJim Jagielski double GridWindow::findMinY()
129*b1cdbd2cSJim Jagielski {
130*b1cdbd2cSJim Jagielski 	if( ! m_pNewYValues )
131*b1cdbd2cSJim Jagielski 		return 0.0;
132*b1cdbd2cSJim Jagielski 	double fMin = m_pNewYValues[0];
133*b1cdbd2cSJim Jagielski 	for( int i = 1; i < m_nValues; i++ )
134*b1cdbd2cSJim Jagielski 		if( m_pNewYValues[ i ] < fMin )
135*b1cdbd2cSJim Jagielski 			fMin = m_pNewYValues[ i ];
136*b1cdbd2cSJim Jagielski 	return fMin;
137*b1cdbd2cSJim Jagielski }
138*b1cdbd2cSJim Jagielski 
139*b1cdbd2cSJim Jagielski // ---------------------------------------------------------------------
140*b1cdbd2cSJim Jagielski 
findMaxX()141*b1cdbd2cSJim Jagielski double GridWindow::findMaxX()
142*b1cdbd2cSJim Jagielski {
143*b1cdbd2cSJim Jagielski 	if( ! m_pXValues )
144*b1cdbd2cSJim Jagielski 		return 0.0;
145*b1cdbd2cSJim Jagielski 	double fMax = m_pXValues[0];
146*b1cdbd2cSJim Jagielski 	for( int i = 1; i < m_nValues; i++ )
147*b1cdbd2cSJim Jagielski 		if( m_pXValues[ i ] > fMax )
148*b1cdbd2cSJim Jagielski 			fMax = m_pXValues[ i ];
149*b1cdbd2cSJim Jagielski 	return fMax;
150*b1cdbd2cSJim Jagielski }
151*b1cdbd2cSJim Jagielski 
152*b1cdbd2cSJim Jagielski // ---------------------------------------------------------------------
153*b1cdbd2cSJim Jagielski 
findMaxY()154*b1cdbd2cSJim Jagielski double GridWindow::findMaxY()
155*b1cdbd2cSJim Jagielski {
156*b1cdbd2cSJim Jagielski 	if( ! m_pNewYValues )
157*b1cdbd2cSJim Jagielski 		return 0.0;
158*b1cdbd2cSJim Jagielski 	double fMax = m_pNewYValues[0];
159*b1cdbd2cSJim Jagielski 	for( int i = 1; i < m_nValues; i++ )
160*b1cdbd2cSJim Jagielski 		if( m_pNewYValues[ i ] > fMax )
161*b1cdbd2cSJim Jagielski 			fMax = m_pNewYValues[ i ];
162*b1cdbd2cSJim Jagielski 	return fMax;
163*b1cdbd2cSJim Jagielski }
164*b1cdbd2cSJim Jagielski 
165*b1cdbd2cSJim Jagielski // ---------------------------------------------------------------------
166*b1cdbd2cSJim Jagielski 
computeExtremes()167*b1cdbd2cSJim Jagielski void GridWindow::computeExtremes()
168*b1cdbd2cSJim Jagielski {
169*b1cdbd2cSJim Jagielski 	if( m_nValues && m_pXValues && m_pOrigYValues )
170*b1cdbd2cSJim Jagielski 	{
171*b1cdbd2cSJim Jagielski 		m_fMaxX = m_fMinX = m_pXValues[0];
172*b1cdbd2cSJim Jagielski 		m_fMaxY = m_fMinY = m_pOrigYValues[0];
173*b1cdbd2cSJim Jagielski 		for( int i = 1; i < m_nValues; i++ )
174*b1cdbd2cSJim Jagielski 		{
175*b1cdbd2cSJim Jagielski 			if( m_pXValues[ i ] > m_fMaxX )
176*b1cdbd2cSJim Jagielski 				m_fMaxX = m_pXValues[ i ];
177*b1cdbd2cSJim Jagielski 			else if( m_pXValues[ i ] < m_fMinX )
178*b1cdbd2cSJim Jagielski 				m_fMinX = m_pXValues[ i ];
179*b1cdbd2cSJim Jagielski 			if( m_pOrigYValues[ i ] > m_fMaxY )
180*b1cdbd2cSJim Jagielski 				m_fMaxY = m_pOrigYValues[ i ];
181*b1cdbd2cSJim Jagielski 			else if( m_pOrigYValues[ i ] < m_fMinY )
182*b1cdbd2cSJim Jagielski 				m_fMinY = m_pOrigYValues[ i ];
183*b1cdbd2cSJim Jagielski 		}
184*b1cdbd2cSJim Jagielski 		setBoundings( m_fMinX, m_fMinY, m_fMaxX, m_fMaxY );
185*b1cdbd2cSJim Jagielski 	}
186*b1cdbd2cSJim Jagielski }
187*b1cdbd2cSJim Jagielski 
188*b1cdbd2cSJim Jagielski // ---------------------------------------------------------------------
189*b1cdbd2cSJim Jagielski 
transform(double x,double y)190*b1cdbd2cSJim Jagielski Point GridWindow::transform( double x, double y )
191*b1cdbd2cSJim Jagielski {
192*b1cdbd2cSJim Jagielski 	Point aRet;
193*b1cdbd2cSJim Jagielski 
194*b1cdbd2cSJim Jagielski 	aRet.X() = (long)( ( x - m_fMinX ) *
195*b1cdbd2cSJim Jagielski 		(double)m_aGridArea.GetWidth() / ( m_fMaxX - m_fMinX )
196*b1cdbd2cSJim Jagielski 		+ m_aGridArea.Left() );
197*b1cdbd2cSJim Jagielski 	aRet.Y() = (long)(
198*b1cdbd2cSJim Jagielski 		m_aGridArea.Bottom() -
199*b1cdbd2cSJim Jagielski 		( y - m_fMinY ) *
200*b1cdbd2cSJim Jagielski 		(double)m_aGridArea.GetHeight() / ( m_fMaxY - m_fMinY ) );
201*b1cdbd2cSJim Jagielski 	return aRet;
202*b1cdbd2cSJim Jagielski }
203*b1cdbd2cSJim Jagielski 
204*b1cdbd2cSJim Jagielski // ---------------------------------------------------------------------
205*b1cdbd2cSJim Jagielski 
transform(const Point & rOriginal,double & x,double & y)206*b1cdbd2cSJim Jagielski void GridWindow::transform( const Point& rOriginal, double& x, double& y )
207*b1cdbd2cSJim Jagielski {
208*b1cdbd2cSJim Jagielski 	x = ( rOriginal.X() - m_aGridArea.Left() ) * (m_fMaxX - m_fMinX) / (double)m_aGridArea.GetWidth() + m_fMinX;
209*b1cdbd2cSJim Jagielski 	y = ( m_aGridArea.Bottom() - rOriginal.Y() ) * (m_fMaxY - m_fMinY) / (double)m_aGridArea.GetHeight() + m_fMinY;
210*b1cdbd2cSJim Jagielski }
211*b1cdbd2cSJim Jagielski 
212*b1cdbd2cSJim Jagielski // ---------------------------------------------------------------------
213*b1cdbd2cSJim Jagielski 
drawLine(double x1,double y1,double x2,double y2)214*b1cdbd2cSJim Jagielski void GridWindow::drawLine( double x1, double y1, double x2, double y2 )
215*b1cdbd2cSJim Jagielski {
216*b1cdbd2cSJim Jagielski 	DrawLine( transform( x1, y1 ), transform( x2, y2 ) );
217*b1cdbd2cSJim Jagielski }
218*b1cdbd2cSJim Jagielski 
219*b1cdbd2cSJim Jagielski // ---------------------------------------------------------------------
220*b1cdbd2cSJim Jagielski 
computeChunk(double fMin,double fMax,double & fChunkOut,double & fMinChunkOut)221*b1cdbd2cSJim Jagielski void GridWindow::computeChunk( double fMin, double fMax, double& fChunkOut, double& fMinChunkOut )
222*b1cdbd2cSJim Jagielski {
223*b1cdbd2cSJim Jagielski 	// get a nice chunk size like 10, 100, 25 or such
224*b1cdbd2cSJim Jagielski 	fChunkOut = ( fMax - fMin ) / 6.0;
225*b1cdbd2cSJim Jagielski 	int logchunk = (int)std::log10( fChunkOut );
226*b1cdbd2cSJim Jagielski 	int nChunk = (int)( fChunkOut / std::exp( (double)(logchunk-1) * M_LN10 ) );
227*b1cdbd2cSJim Jagielski 	if( nChunk >= 75 )
228*b1cdbd2cSJim Jagielski 		nChunk = 100;
229*b1cdbd2cSJim Jagielski 	else if( nChunk >= 35 )
230*b1cdbd2cSJim Jagielski 		nChunk = 50;
231*b1cdbd2cSJim Jagielski 	else if ( nChunk > 20 )
232*b1cdbd2cSJim Jagielski 		nChunk = 25;
233*b1cdbd2cSJim Jagielski 	else if ( nChunk >= 13 )
234*b1cdbd2cSJim Jagielski 		nChunk = 20;
235*b1cdbd2cSJim Jagielski 	else if( nChunk > 5 )
236*b1cdbd2cSJim Jagielski 		nChunk = 10;
237*b1cdbd2cSJim Jagielski 	else
238*b1cdbd2cSJim Jagielski 		nChunk = 5;
239*b1cdbd2cSJim Jagielski 	fChunkOut = (double) nChunk * exp( (double)(logchunk-1) * M_LN10 );
240*b1cdbd2cSJim Jagielski 	// compute whole chunks fitting into fMin
241*b1cdbd2cSJim Jagielski 	nChunk = (int)( fMin / fChunkOut );
242*b1cdbd2cSJim Jagielski 	fMinChunkOut = (double)nChunk * fChunkOut;
243*b1cdbd2cSJim Jagielski 	while( fMinChunkOut < fMin )
244*b1cdbd2cSJim Jagielski 		fMinChunkOut += fChunkOut;
245*b1cdbd2cSJim Jagielski }
246*b1cdbd2cSJim Jagielski 
247*b1cdbd2cSJim Jagielski // ---------------------------------------------------------------------
248*b1cdbd2cSJim Jagielski 
computeNew()249*b1cdbd2cSJim Jagielski void GridWindow::computeNew()
250*b1cdbd2cSJim Jagielski {
251*b1cdbd2cSJim Jagielski 	if(2L == m_aHandles.size())
252*b1cdbd2cSJim Jagielski 	{
253*b1cdbd2cSJim Jagielski 		// special case: only left and right markers
254*b1cdbd2cSJim Jagielski 		double xleft, yleft;
255*b1cdbd2cSJim Jagielski 		double xright, yright;
256*b1cdbd2cSJim Jagielski 		transform(m_aHandles[0L].maPos, xleft, yleft);
257*b1cdbd2cSJim Jagielski 		transform(m_aHandles[1L].maPos, xright, yright );
258*b1cdbd2cSJim Jagielski 		double factor = (yright-yleft)/(xright-xleft);
259*b1cdbd2cSJim Jagielski 		for( int i = 0; i < m_nValues; i++ )
260*b1cdbd2cSJim Jagielski 		{
261*b1cdbd2cSJim Jagielski 			m_pNewYValues[ i ] = yleft + ( m_pXValues[ i ] - xleft )*factor;
262*b1cdbd2cSJim Jagielski 		}
263*b1cdbd2cSJim Jagielski 	}
264*b1cdbd2cSJim Jagielski 	else
265*b1cdbd2cSJim Jagielski 	{
266*b1cdbd2cSJim Jagielski 		// sort markers
267*b1cdbd2cSJim Jagielski 		std::sort(m_aHandles.begin(), m_aHandles.end());
268*b1cdbd2cSJim Jagielski 		const int nSorted = m_aHandles.size();
269*b1cdbd2cSJim Jagielski 		int i;
270*b1cdbd2cSJim Jagielski 
271*b1cdbd2cSJim Jagielski 		// get node arrays
272*b1cdbd2cSJim Jagielski 		double* nodex = new double[ nSorted ];
273*b1cdbd2cSJim Jagielski 		double* nodey = new double[ nSorted ];
274*b1cdbd2cSJim Jagielski 
275*b1cdbd2cSJim Jagielski 		for( i = 0L; i < nSorted; i++ )
276*b1cdbd2cSJim Jagielski 			transform( m_aHandles[i].maPos, nodex[ i ], nodey[ i ] );
277*b1cdbd2cSJim Jagielski 
278*b1cdbd2cSJim Jagielski 		for( i = 0; i < m_nValues; i++ )
279*b1cdbd2cSJim Jagielski 		{
280*b1cdbd2cSJim Jagielski 			double x = m_pXValues[ i ];
281*b1cdbd2cSJim Jagielski 			m_pNewYValues[ i ] = interpolate( x, nodex, nodey, nSorted );
282*b1cdbd2cSJim Jagielski 			if( m_bCutValues )
283*b1cdbd2cSJim Jagielski 			{
284*b1cdbd2cSJim Jagielski 				if( m_pNewYValues[ i ] > m_fMaxY )
285*b1cdbd2cSJim Jagielski 					m_pNewYValues[ i ] = m_fMaxY;
286*b1cdbd2cSJim Jagielski 				else if( m_pNewYValues[ i ] < m_fMinY )
287*b1cdbd2cSJim Jagielski 					m_pNewYValues[ i ] = m_fMinY;
288*b1cdbd2cSJim Jagielski 			}
289*b1cdbd2cSJim Jagielski 		}
290*b1cdbd2cSJim Jagielski 
291*b1cdbd2cSJim Jagielski 		delete [] nodex;
292*b1cdbd2cSJim Jagielski 		delete [] nodey;
293*b1cdbd2cSJim Jagielski 	}
294*b1cdbd2cSJim Jagielski }
295*b1cdbd2cSJim Jagielski 
296*b1cdbd2cSJim Jagielski // ---------------------------------------------------------------------
297*b1cdbd2cSJim Jagielski 
interpolate(double x,double * pNodeX,double * pNodeY,int nNodes)298*b1cdbd2cSJim Jagielski double GridWindow::interpolate(
299*b1cdbd2cSJim Jagielski 	double x,
300*b1cdbd2cSJim Jagielski 	double* pNodeX,
301*b1cdbd2cSJim Jagielski 	double* pNodeY,
302*b1cdbd2cSJim Jagielski 	int nNodes )
303*b1cdbd2cSJim Jagielski {
304*b1cdbd2cSJim Jagielski 	// compute Lagrange interpolation
305*b1cdbd2cSJim Jagielski 	double ret = 0;
306*b1cdbd2cSJim Jagielski 	for( int i = 0; i < nNodes; i++ )
307*b1cdbd2cSJim Jagielski 	{
308*b1cdbd2cSJim Jagielski 		double sum = pNodeY[ i ];
309*b1cdbd2cSJim Jagielski 		for( int n = 0; n < nNodes; n++ )
310*b1cdbd2cSJim Jagielski 		{
311*b1cdbd2cSJim Jagielski 			if( n != i )
312*b1cdbd2cSJim Jagielski 			{
313*b1cdbd2cSJim Jagielski 				sum *= x - pNodeX[ n ];
314*b1cdbd2cSJim Jagielski 				sum /= pNodeX[ i ] - pNodeX[ n ];
315*b1cdbd2cSJim Jagielski 			}
316*b1cdbd2cSJim Jagielski 		}
317*b1cdbd2cSJim Jagielski 		ret += sum;
318*b1cdbd2cSJim Jagielski 	}
319*b1cdbd2cSJim Jagielski 	return ret;
320*b1cdbd2cSJim Jagielski }
321*b1cdbd2cSJim Jagielski 
322*b1cdbd2cSJim Jagielski // ---------------------------------------------------------------------
323*b1cdbd2cSJim Jagielski 
setBoundings(double fMinX,double fMinY,double fMaxX,double fMaxY)324*b1cdbd2cSJim Jagielski void GridWindow::setBoundings( double fMinX, double fMinY, double fMaxX, double fMaxY )
325*b1cdbd2cSJim Jagielski {
326*b1cdbd2cSJim Jagielski 	m_fMinX = fMinX;
327*b1cdbd2cSJim Jagielski 	m_fMinY = fMinY;
328*b1cdbd2cSJim Jagielski 	m_fMaxX = fMaxX;
329*b1cdbd2cSJim Jagielski 	m_fMaxY = fMaxY;
330*b1cdbd2cSJim Jagielski 
331*b1cdbd2cSJim Jagielski 	computeChunk( m_fMinX, m_fMaxX, m_fChunkX, m_fMinChunkX );
332*b1cdbd2cSJim Jagielski 	computeChunk( m_fMinY, m_fMaxY, m_fChunkY, m_fMinChunkY );
333*b1cdbd2cSJim Jagielski }
334*b1cdbd2cSJim Jagielski 
335*b1cdbd2cSJim Jagielski // ---------------------------------------------------------------------
336*b1cdbd2cSJim Jagielski 
drawGrid()337*b1cdbd2cSJim Jagielski void GridWindow::drawGrid()
338*b1cdbd2cSJim Jagielski {
339*b1cdbd2cSJim Jagielski 	char pBuf[256];
340*b1cdbd2cSJim Jagielski 	SetLineColor( Color( COL_BLACK ) );
341*b1cdbd2cSJim Jagielski 	// draw vertical lines
342*b1cdbd2cSJim Jagielski 	for( double fX = m_fMinChunkX; fX < m_fMaxX; fX += m_fChunkX )
343*b1cdbd2cSJim Jagielski 	{
344*b1cdbd2cSJim Jagielski 		drawLine( fX, m_fMinY, fX, m_fMaxY );
345*b1cdbd2cSJim Jagielski 		// draw tickmarks
346*b1cdbd2cSJim Jagielski 		Point aPt = transform( fX, m_fMinY );
347*b1cdbd2cSJim Jagielski         std::sprintf( pBuf, "%g", fX );
348*b1cdbd2cSJim Jagielski 		String aMark( pBuf, gsl_getSystemTextEncoding() );
349*b1cdbd2cSJim Jagielski 		Size aTextSize( GetTextWidth( aMark ), GetTextHeight() );
350*b1cdbd2cSJim Jagielski 		aPt.X() -= aTextSize.Width()/2;
351*b1cdbd2cSJim Jagielski 		aPt.Y() += aTextSize.Height()/2;
352*b1cdbd2cSJim Jagielski 		DrawText( aPt, aMark );
353*b1cdbd2cSJim Jagielski 	}
354*b1cdbd2cSJim Jagielski 	// draw horizontal lines
355*b1cdbd2cSJim Jagielski 	for( double fY = m_fMinChunkY; fY < m_fMaxY; fY += m_fChunkY )
356*b1cdbd2cSJim Jagielski 	{
357*b1cdbd2cSJim Jagielski 		drawLine( m_fMinX, fY, m_fMaxX, fY );
358*b1cdbd2cSJim Jagielski 		// draw tickmarks
359*b1cdbd2cSJim Jagielski 		Point aPt = transform( m_fMinX, fY );
360*b1cdbd2cSJim Jagielski         std::sprintf( pBuf, "%g", fY );
361*b1cdbd2cSJim Jagielski 		String aMark( pBuf, gsl_getSystemTextEncoding() );
362*b1cdbd2cSJim Jagielski 		Size aTextSize( GetTextWidth( aMark ), GetTextHeight() );
363*b1cdbd2cSJim Jagielski 		aPt.X() -= aTextSize.Width() + 2;
364*b1cdbd2cSJim Jagielski 		aPt.Y() -= aTextSize.Height()/2;
365*b1cdbd2cSJim Jagielski 		DrawText( aPt, aMark );
366*b1cdbd2cSJim Jagielski 	}
367*b1cdbd2cSJim Jagielski 
368*b1cdbd2cSJim Jagielski 	// draw boundings
369*b1cdbd2cSJim Jagielski 	drawLine( m_fMinX, m_fMinY, m_fMaxX, m_fMinY );
370*b1cdbd2cSJim Jagielski 	drawLine( m_fMinX, m_fMaxY, m_fMaxX, m_fMaxY );
371*b1cdbd2cSJim Jagielski 	drawLine( m_fMinX, m_fMinY, m_fMinX, m_fMaxY );
372*b1cdbd2cSJim Jagielski 	drawLine( m_fMaxX, m_fMinY, m_fMaxX, m_fMaxY );
373*b1cdbd2cSJim Jagielski }
374*b1cdbd2cSJim Jagielski 
375*b1cdbd2cSJim Jagielski // ---------------------------------------------------------------------
376*b1cdbd2cSJim Jagielski 
drawOriginal()377*b1cdbd2cSJim Jagielski void GridWindow::drawOriginal()
378*b1cdbd2cSJim Jagielski {
379*b1cdbd2cSJim Jagielski 	if( m_nValues && m_pXValues && m_pOrigYValues )
380*b1cdbd2cSJim Jagielski 	{
381*b1cdbd2cSJim Jagielski 		SetLineColor( Color( COL_RED ) );
382*b1cdbd2cSJim Jagielski 		for( int i = 0; i < m_nValues-1; i++ )
383*b1cdbd2cSJim Jagielski 		{
384*b1cdbd2cSJim Jagielski 			drawLine( m_pXValues[ i   ], m_pOrigYValues[ i   ],
385*b1cdbd2cSJim Jagielski 					  m_pXValues[ i+1 ], m_pOrigYValues[ i+1 ] );
386*b1cdbd2cSJim Jagielski 		}
387*b1cdbd2cSJim Jagielski 	}
388*b1cdbd2cSJim Jagielski }
389*b1cdbd2cSJim Jagielski 
390*b1cdbd2cSJim Jagielski // ---------------------------------------------------------------------
391*b1cdbd2cSJim Jagielski 
drawNew()392*b1cdbd2cSJim Jagielski void GridWindow::drawNew()
393*b1cdbd2cSJim Jagielski {
394*b1cdbd2cSJim Jagielski 	if( m_nValues && m_pXValues && m_pNewYValues )
395*b1cdbd2cSJim Jagielski 	{
396*b1cdbd2cSJim Jagielski 		SetClipRegion( m_aGridArea );
397*b1cdbd2cSJim Jagielski 		SetLineColor( Color( COL_YELLOW ) );
398*b1cdbd2cSJim Jagielski 		for( int i = 0; i < m_nValues-1; i++ )
399*b1cdbd2cSJim Jagielski 		{
400*b1cdbd2cSJim Jagielski 			drawLine( m_pXValues[ i   ], m_pNewYValues[ i   ],
401*b1cdbd2cSJim Jagielski 					  m_pXValues[ i+1 ], m_pNewYValues[ i+1 ] );
402*b1cdbd2cSJim Jagielski 		}
403*b1cdbd2cSJim Jagielski 		SetClipRegion();
404*b1cdbd2cSJim Jagielski 	}
405*b1cdbd2cSJim Jagielski }
406*b1cdbd2cSJim Jagielski 
407*b1cdbd2cSJim Jagielski // ---------------------------------------------------------------------
408*b1cdbd2cSJim Jagielski 
drawHandles()409*b1cdbd2cSJim Jagielski void GridWindow::drawHandles()
410*b1cdbd2cSJim Jagielski {
411*b1cdbd2cSJim Jagielski 	for(sal_uInt32 i(0L); i < m_aHandles.size(); i++)
412*b1cdbd2cSJim Jagielski 	{
413*b1cdbd2cSJim Jagielski 		m_aHandles[i].draw(*this, m_aMarkerBitmap);
414*b1cdbd2cSJim Jagielski 	}
415*b1cdbd2cSJim Jagielski }
416*b1cdbd2cSJim Jagielski 
417*b1cdbd2cSJim Jagielski // ---------------------------------------------------------------------
418*b1cdbd2cSJim Jagielski 
Paint(const Rectangle & rRect)419*b1cdbd2cSJim Jagielski void GridWindow::Paint( const Rectangle& rRect )
420*b1cdbd2cSJim Jagielski {
421*b1cdbd2cSJim Jagielski 	ModalDialog::Paint( rRect );
422*b1cdbd2cSJim Jagielski 	drawGrid();
423*b1cdbd2cSJim Jagielski 	drawOriginal();
424*b1cdbd2cSJim Jagielski 	drawNew();
425*b1cdbd2cSJim Jagielski 	drawHandles();
426*b1cdbd2cSJim Jagielski }
427*b1cdbd2cSJim Jagielski 
428*b1cdbd2cSJim Jagielski // ---------------------------------------------------------------------
429*b1cdbd2cSJim Jagielski 
MouseMove(const MouseEvent & rEvt)430*b1cdbd2cSJim Jagielski void GridWindow::MouseMove( const MouseEvent& rEvt )
431*b1cdbd2cSJim Jagielski {
432*b1cdbd2cSJim Jagielski 	if( rEvt.GetButtons() == MOUSE_LEFT && m_nDragIndex != 0xffffffff )
433*b1cdbd2cSJim Jagielski 	{
434*b1cdbd2cSJim Jagielski 		Point aPoint( rEvt.GetPosPixel() );
435*b1cdbd2cSJim Jagielski 
436*b1cdbd2cSJim Jagielski 		if( m_nDragIndex == 0L || m_nDragIndex == m_aHandles.size() - 1L)
437*b1cdbd2cSJim Jagielski 		{
438*b1cdbd2cSJim Jagielski 			aPoint.X() = m_aHandles[m_nDragIndex].maPos.X();
439*b1cdbd2cSJim Jagielski 		}
440*b1cdbd2cSJim Jagielski 		else
441*b1cdbd2cSJim Jagielski 		{
442*b1cdbd2cSJim Jagielski 			if(aPoint.X() < m_aGridArea.Left())
443*b1cdbd2cSJim Jagielski 				aPoint.X() = m_aGridArea.Left();
444*b1cdbd2cSJim Jagielski 			else if(aPoint.X() > m_aGridArea.Right())
445*b1cdbd2cSJim Jagielski 				aPoint.X() = m_aGridArea.Right();
446*b1cdbd2cSJim Jagielski 		}
447*b1cdbd2cSJim Jagielski 
448*b1cdbd2cSJim Jagielski 		if( aPoint.Y() < m_aGridArea.Top() )
449*b1cdbd2cSJim Jagielski 			aPoint.Y() = m_aGridArea.Top();
450*b1cdbd2cSJim Jagielski 		else if( aPoint.Y() > m_aGridArea.Bottom() )
451*b1cdbd2cSJim Jagielski 			aPoint.Y() = m_aGridArea.Bottom();
452*b1cdbd2cSJim Jagielski 
453*b1cdbd2cSJim Jagielski 		if( aPoint != m_aHandles[m_nDragIndex].maPos )
454*b1cdbd2cSJim Jagielski 		{
455*b1cdbd2cSJim Jagielski 			m_aHandles[m_nDragIndex].maPos = aPoint;
456*b1cdbd2cSJim Jagielski 			Invalidate( m_aGridArea );
457*b1cdbd2cSJim Jagielski 		}
458*b1cdbd2cSJim Jagielski 	}
459*b1cdbd2cSJim Jagielski 
460*b1cdbd2cSJim Jagielski 	ModalDialog::MouseMove( rEvt );
461*b1cdbd2cSJim Jagielski }
462*b1cdbd2cSJim Jagielski 
463*b1cdbd2cSJim Jagielski // ---------------------------------------------------------------------
464*b1cdbd2cSJim Jagielski 
MouseButtonUp(const MouseEvent & rEvt)465*b1cdbd2cSJim Jagielski void GridWindow::MouseButtonUp( const MouseEvent& rEvt )
466*b1cdbd2cSJim Jagielski {
467*b1cdbd2cSJim Jagielski 	if( rEvt.GetButtons() == MOUSE_LEFT )
468*b1cdbd2cSJim Jagielski 	{
469*b1cdbd2cSJim Jagielski 		if( m_nDragIndex != 0xffffffff )
470*b1cdbd2cSJim Jagielski 		{
471*b1cdbd2cSJim Jagielski 			m_nDragIndex = 0xffffffff;
472*b1cdbd2cSJim Jagielski 			computeNew();
473*b1cdbd2cSJim Jagielski 			Invalidate( m_aGridArea );
474*b1cdbd2cSJim Jagielski 			Paint( m_aGridArea );
475*b1cdbd2cSJim Jagielski 		}
476*b1cdbd2cSJim Jagielski 	}
477*b1cdbd2cSJim Jagielski 
478*b1cdbd2cSJim Jagielski 	ModalDialog::MouseButtonUp( rEvt );
479*b1cdbd2cSJim Jagielski }
480*b1cdbd2cSJim Jagielski 
481*b1cdbd2cSJim Jagielski // ---------------------------------------------------------------------
482*b1cdbd2cSJim Jagielski 
MouseButtonDown(const MouseEvent & rEvt)483*b1cdbd2cSJim Jagielski void GridWindow::MouseButtonDown( const MouseEvent& rEvt )
484*b1cdbd2cSJim Jagielski {
485*b1cdbd2cSJim Jagielski 	Point aPoint( rEvt.GetPosPixel() );
486*b1cdbd2cSJim Jagielski 	sal_uInt32 nMarkerIndex = 0xffffffff;
487*b1cdbd2cSJim Jagielski 
488*b1cdbd2cSJim Jagielski 	for(sal_uInt32 a(0L); nMarkerIndex == 0xffffffff && a < m_aHandles.size(); a++)
489*b1cdbd2cSJim Jagielski 	{
490*b1cdbd2cSJim Jagielski 		if(m_aHandles[a].isHit(*this, aPoint))
491*b1cdbd2cSJim Jagielski 		{
492*b1cdbd2cSJim Jagielski 			nMarkerIndex = a;
493*b1cdbd2cSJim Jagielski 		}
494*b1cdbd2cSJim Jagielski 	}
495*b1cdbd2cSJim Jagielski 
496*b1cdbd2cSJim Jagielski 	if( rEvt.GetButtons() == MOUSE_LEFT )
497*b1cdbd2cSJim Jagielski 	{
498*b1cdbd2cSJim Jagielski 		// user wants to drag a button
499*b1cdbd2cSJim Jagielski 		if( nMarkerIndex != 0xffffffff )
500*b1cdbd2cSJim Jagielski 		{
501*b1cdbd2cSJim Jagielski 			m_nDragIndex = nMarkerIndex;
502*b1cdbd2cSJim Jagielski 		}
503*b1cdbd2cSJim Jagielski 	}
504*b1cdbd2cSJim Jagielski 	else if( rEvt.GetButtons() == MOUSE_RIGHT )
505*b1cdbd2cSJim Jagielski 	{
506*b1cdbd2cSJim Jagielski 		// user wants to add/delete a button
507*b1cdbd2cSJim Jagielski 		if( nMarkerIndex != 0xffffffff )
508*b1cdbd2cSJim Jagielski 		{
509*b1cdbd2cSJim Jagielski 			if( nMarkerIndex != 0L && nMarkerIndex != m_aHandles.size() - 1L)
510*b1cdbd2cSJim Jagielski 			{
511*b1cdbd2cSJim Jagielski 				// delete marker under mouse
512*b1cdbd2cSJim Jagielski 				if( m_nDragIndex == nMarkerIndex )
513*b1cdbd2cSJim Jagielski 					m_nDragIndex = 0xffffffff;
514*b1cdbd2cSJim Jagielski 
515*b1cdbd2cSJim Jagielski 				m_aHandles.erase(m_aHandles.begin() + nMarkerIndex);
516*b1cdbd2cSJim Jagielski 			}
517*b1cdbd2cSJim Jagielski 		}
518*b1cdbd2cSJim Jagielski 		else
519*b1cdbd2cSJim Jagielski 		{
520*b1cdbd2cSJim Jagielski 			m_BmOffX = sal_uInt16(m_aMarkerBitmap.GetSizePixel().Width() >> 1);
521*b1cdbd2cSJim Jagielski 			m_BmOffY = sal_uInt16(m_aMarkerBitmap.GetSizePixel().Height() >> 1);
522*b1cdbd2cSJim Jagielski 			m_aHandles.push_back(impHandle(aPoint, m_BmOffX, m_BmOffY));
523*b1cdbd2cSJim Jagielski 		}
524*b1cdbd2cSJim Jagielski 
525*b1cdbd2cSJim Jagielski 		computeNew();
526*b1cdbd2cSJim Jagielski 		Invalidate( m_aGridArea );
527*b1cdbd2cSJim Jagielski 		Paint( m_aGridArea );
528*b1cdbd2cSJim Jagielski 	}
529*b1cdbd2cSJim Jagielski 
530*b1cdbd2cSJim Jagielski 	ModalDialog::MouseButtonDown( rEvt );
531*b1cdbd2cSJim Jagielski }
532*b1cdbd2cSJim Jagielski 
533*b1cdbd2cSJim Jagielski // ---------------------------------------------------------------------
534*b1cdbd2cSJim Jagielski 
IMPL_LINK(GridWindow,ClickButtonHdl,Button *,pButton)535*b1cdbd2cSJim Jagielski IMPL_LINK( GridWindow, ClickButtonHdl, Button*, pButton )
536*b1cdbd2cSJim Jagielski {
537*b1cdbd2cSJim Jagielski 	if( pButton == &m_aResetButton )
538*b1cdbd2cSJim Jagielski 	{
539*b1cdbd2cSJim Jagielski 		int nType = (int)(sal_IntPtr)m_aResetTypeBox.GetEntryData( m_aResetTypeBox.GetSelectEntryPos() );
540*b1cdbd2cSJim Jagielski 		switch( nType )
541*b1cdbd2cSJim Jagielski 		{
542*b1cdbd2cSJim Jagielski 			case RESET_TYPE_LINEAR_ASCENDING:
543*b1cdbd2cSJim Jagielski 			{
544*b1cdbd2cSJim Jagielski 				for( int i = 0; i < m_nValues; i++ )
545*b1cdbd2cSJim Jagielski 				{
546*b1cdbd2cSJim Jagielski 					m_pNewYValues[ i ] = m_fMinY + (m_fMaxY-m_fMinY)/(m_fMaxX-m_fMinX)*(m_pXValues[i]-m_fMinX);
547*b1cdbd2cSJim Jagielski 				}
548*b1cdbd2cSJim Jagielski 			}
549*b1cdbd2cSJim Jagielski 			break;
550*b1cdbd2cSJim Jagielski 			case RESET_TYPE_LINEAR_DESCENDING:
551*b1cdbd2cSJim Jagielski 			{
552*b1cdbd2cSJim Jagielski 				for( int i = 0; i < m_nValues; i++ )
553*b1cdbd2cSJim Jagielski 				{
554*b1cdbd2cSJim Jagielski 					m_pNewYValues[ i ] = m_fMaxY - (m_fMaxY-m_fMinY)/(m_fMaxX-m_fMinX)*(m_pXValues[i]-m_fMinX);
555*b1cdbd2cSJim Jagielski 				}
556*b1cdbd2cSJim Jagielski 			}
557*b1cdbd2cSJim Jagielski 			break;
558*b1cdbd2cSJim Jagielski 			case RESET_TYPE_RESET:
559*b1cdbd2cSJim Jagielski 			{
560*b1cdbd2cSJim Jagielski 				if( m_pOrigYValues && m_pNewYValues && m_nValues )
561*b1cdbd2cSJim Jagielski 					memcpy( m_pNewYValues, m_pOrigYValues, m_nValues*sizeof(double) );
562*b1cdbd2cSJim Jagielski 			}
563*b1cdbd2cSJim Jagielski 			break;
564*b1cdbd2cSJim Jagielski 			case RESET_TYPE_EXPONENTIAL:
565*b1cdbd2cSJim Jagielski 			{
566*b1cdbd2cSJim Jagielski 				for( int i = 0; i < m_nValues; i++ )
567*b1cdbd2cSJim Jagielski 				{
568*b1cdbd2cSJim Jagielski 					m_pNewYValues[ i ] = m_fMinY + (m_fMaxY-m_fMinY)*(std::exp((m_pXValues[i]-m_fMinX)/(m_fMaxX-m_fMinX))-1.0)/(M_E-1.0);
569*b1cdbd2cSJim Jagielski 				}
570*b1cdbd2cSJim Jagielski 			}
571*b1cdbd2cSJim Jagielski 			break;
572*b1cdbd2cSJim Jagielski 
573*b1cdbd2cSJim Jagielski 			default:
574*b1cdbd2cSJim Jagielski 				break;
575*b1cdbd2cSJim Jagielski 		}
576*b1cdbd2cSJim Jagielski 
577*b1cdbd2cSJim Jagielski 		for(sal_uInt32 i(0L); i < m_aHandles.size(); i++)
578*b1cdbd2cSJim Jagielski 		{
579*b1cdbd2cSJim Jagielski 			// find nearest xvalue
580*b1cdbd2cSJim Jagielski 			double x, y;
581*b1cdbd2cSJim Jagielski 			transform( m_aHandles[i].maPos, x, y );
582*b1cdbd2cSJim Jagielski 			int nIndex = 0;
583*b1cdbd2cSJim Jagielski 			double delta = std::fabs( x-m_pXValues[0] );
584*b1cdbd2cSJim Jagielski 			for( int n = 1; n < m_nValues; n++ )
585*b1cdbd2cSJim Jagielski 			{
586*b1cdbd2cSJim Jagielski 				if( delta > std::fabs( x - m_pXValues[ n ] ) )
587*b1cdbd2cSJim Jagielski 				{
588*b1cdbd2cSJim Jagielski 					delta = std::fabs( x - m_pXValues[ n ] );
589*b1cdbd2cSJim Jagielski 					nIndex = n;
590*b1cdbd2cSJim Jagielski 				}
591*b1cdbd2cSJim Jagielski 			}
592*b1cdbd2cSJim Jagielski 			if( 0 == i )
593*b1cdbd2cSJim Jagielski 				m_aHandles[i].maPos = transform( m_fMinX, m_pNewYValues[ nIndex ] );
594*b1cdbd2cSJim Jagielski 			else if( m_aHandles.size() - 1L == i )
595*b1cdbd2cSJim Jagielski 				m_aHandles[i].maPos = transform( m_fMaxX, m_pNewYValues[ nIndex ] );
596*b1cdbd2cSJim Jagielski 			else
597*b1cdbd2cSJim Jagielski 				m_aHandles[i].maPos = transform( m_pXValues[ nIndex ], m_pNewYValues[ nIndex ] );
598*b1cdbd2cSJim Jagielski 		}
599*b1cdbd2cSJim Jagielski 
600*b1cdbd2cSJim Jagielski 		Invalidate( m_aGridArea );
601*b1cdbd2cSJim Jagielski 		Paint(Rectangle());
602*b1cdbd2cSJim Jagielski 	}
603*b1cdbd2cSJim Jagielski 	return 0;
604*b1cdbd2cSJim Jagielski }
605