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_xmlhelp.hxx"
26*b1cdbd2cSJim Jagielski #include <com/sun/star/ucb/Command.hpp>
27*b1cdbd2cSJim Jagielski #include <com/sun/star/ucb/XCommandEnvironment.hpp>
28*b1cdbd2cSJim Jagielski #include <com/sun/star/i18n/XExtendedTransliteration.hpp>
29*b1cdbd2cSJim Jagielski #include <com/sun/star/ucb/XCommandProcessor.hpp>
30*b1cdbd2cSJim Jagielski #include <com/sun/star/lang/Locale.hpp>
31*b1cdbd2cSJim Jagielski #include <com/sun/star/script/XInvocation.hpp>
32*b1cdbd2cSJim Jagielski 
33*b1cdbd2cSJim Jagielski #ifndef INCLUDED_STL_ALGORITHM
34*b1cdbd2cSJim Jagielski #include <algorithm>
35*b1cdbd2cSJim Jagielski #define INCLUDED_STL_ALGORITHM
36*b1cdbd2cSJim Jagielski #endif
37*b1cdbd2cSJim Jagielski #ifndef INCLUDED_STL_SET
38*b1cdbd2cSJim Jagielski #include <set>
39*b1cdbd2cSJim Jagielski #define INCLUDED_STL_SET
40*b1cdbd2cSJim Jagielski #endif
41*b1cdbd2cSJim Jagielski 
42*b1cdbd2cSJim Jagielski #include <qe/Query.hxx>
43*b1cdbd2cSJim Jagielski #include <qe/DocGenerator.hxx>
44*b1cdbd2cSJim Jagielski #include "resultsetforquery.hxx"
45*b1cdbd2cSJim Jagielski #include "databases.hxx"
46*b1cdbd2cSJim Jagielski 
47*b1cdbd2cSJim Jagielski // For testing
48*b1cdbd2cSJim Jagielski // #define LOGGING
49*b1cdbd2cSJim Jagielski 
50*b1cdbd2cSJim Jagielski using namespace std;
51*b1cdbd2cSJim Jagielski using namespace chelp;
52*b1cdbd2cSJim Jagielski using namespace xmlsearch::excep;
53*b1cdbd2cSJim Jagielski using namespace xmlsearch::qe;
54*b1cdbd2cSJim Jagielski using namespace com::sun::star;
55*b1cdbd2cSJim Jagielski using namespace com::sun::star::ucb;
56*b1cdbd2cSJim Jagielski using namespace com::sun::star::i18n;
57*b1cdbd2cSJim Jagielski using namespace com::sun::star::uno;
58*b1cdbd2cSJim Jagielski using namespace com::sun::star::lang;
59*b1cdbd2cSJim Jagielski 
60*b1cdbd2cSJim Jagielski struct HitItem
61*b1cdbd2cSJim Jagielski {
62*b1cdbd2cSJim Jagielski 	rtl::OUString	m_aURL;
63*b1cdbd2cSJim Jagielski 	float			m_fScore;
64*b1cdbd2cSJim Jagielski 
HitItemHitItem65*b1cdbd2cSJim Jagielski 	HitItem( void )	{}
HitItemHitItem66*b1cdbd2cSJim Jagielski 	HitItem( const rtl::OUString& aURL, float fScore )
67*b1cdbd2cSJim Jagielski 		: m_aURL( aURL )
68*b1cdbd2cSJim Jagielski 		, m_fScore( fScore )
69*b1cdbd2cSJim Jagielski 	{}
operator <HitItem70*b1cdbd2cSJim Jagielski 	bool operator < ( const HitItem& rHitItem ) const
71*b1cdbd2cSJim Jagielski 	{
72*b1cdbd2cSJim Jagielski 		return rHitItem.m_fScore < m_fScore;
73*b1cdbd2cSJim Jagielski 	}
74*b1cdbd2cSJim Jagielski };
75*b1cdbd2cSJim Jagielski 
ResultSetForQuery(const uno::Reference<lang::XMultiServiceFactory> & xMSF,const uno::Reference<XContentProvider> & xProvider,sal_Int32 nOpenMode,const uno::Sequence<beans::Property> & seq,const uno::Sequence<NumberedSortingInfo> & seqSort,URLParameter & aURLParameter,Databases * pDatabases)76*b1cdbd2cSJim Jagielski ResultSetForQuery::ResultSetForQuery( const uno::Reference< lang::XMultiServiceFactory >&  xMSF,
77*b1cdbd2cSJim Jagielski 									  const uno::Reference< XContentProvider >&  xProvider,
78*b1cdbd2cSJim Jagielski 									  sal_Int32 nOpenMode,
79*b1cdbd2cSJim Jagielski 									  const uno::Sequence< beans::Property >& seq,
80*b1cdbd2cSJim Jagielski 									  const uno::Sequence< NumberedSortingInfo >& seqSort,
81*b1cdbd2cSJim Jagielski 									  URLParameter& aURLParameter,
82*b1cdbd2cSJim Jagielski 									  Databases* pDatabases )
83*b1cdbd2cSJim Jagielski 	: ResultSetBase( xMSF,xProvider,nOpenMode,seq,seqSort ),
84*b1cdbd2cSJim Jagielski 	  m_pDatabases( pDatabases ),
85*b1cdbd2cSJim Jagielski       m_aURLParameter( aURLParameter )
86*b1cdbd2cSJim Jagielski {
87*b1cdbd2cSJim Jagielski     Reference< XTransliteration > xTrans(
88*b1cdbd2cSJim Jagielski         xMSF->createInstance( rtl::OUString::createFromAscii( "com.sun.star.i18n.Transliteration" ) ),
89*b1cdbd2cSJim Jagielski         UNO_QUERY );
90*b1cdbd2cSJim Jagielski     Locale aLocale( aURLParameter.get_language(),
91*b1cdbd2cSJim Jagielski                     rtl::OUString(),
92*b1cdbd2cSJim Jagielski                     rtl::OUString() );
93*b1cdbd2cSJim Jagielski     if(xTrans.is())
94*b1cdbd2cSJim Jagielski         xTrans->loadModule(TransliterationModules_UPPERCASE_LOWERCASE,
95*b1cdbd2cSJim Jagielski                            aLocale );
96*b1cdbd2cSJim Jagielski 
97*b1cdbd2cSJim Jagielski 	// Access Lucene via XInvocation
98*b1cdbd2cSJim Jagielski 	Reference< script::XInvocation > xInvocation(
99*b1cdbd2cSJim Jagielski         xMSF->createInstance( rtl::OUString::createFromAscii( "com.sun.star.help.HelpSearch" ) ),
100*b1cdbd2cSJim Jagielski         UNO_QUERY );
101*b1cdbd2cSJim Jagielski 
102*b1cdbd2cSJim Jagielski 	vector< vector< rtl::OUString > > queryList;
103*b1cdbd2cSJim Jagielski 	{
104*b1cdbd2cSJim Jagielski 		sal_Int32 idx;
105*b1cdbd2cSJim Jagielski 		rtl::OUString query = m_aURLParameter.get_query();
106*b1cdbd2cSJim Jagielski 		while( query.getLength() )
107*b1cdbd2cSJim Jagielski 		{
108*b1cdbd2cSJim Jagielski 			idx = query.indexOf( sal_Unicode( ' ' ) );
109*b1cdbd2cSJim Jagielski 			if( idx == -1 )
110*b1cdbd2cSJim Jagielski 				idx = query.getLength();
111*b1cdbd2cSJim Jagielski 
112*b1cdbd2cSJim Jagielski 			vector< rtl::OUString > currentQuery;
113*b1cdbd2cSJim Jagielski             rtl::OUString tmp(query.copy( 0,idx ));
114*b1cdbd2cSJim Jagielski             rtl:: OUString toliterate = tmp;
115*b1cdbd2cSJim Jagielski             if(xTrans.is()) {
116*b1cdbd2cSJim Jagielski                 Sequence<sal_Int32> aSeq;
117*b1cdbd2cSJim Jagielski                 toliterate = xTrans->transliterate(
118*b1cdbd2cSJim Jagielski                     tmp,0,tmp.getLength(),aSeq);
119*b1cdbd2cSJim Jagielski             }
120*b1cdbd2cSJim Jagielski 
121*b1cdbd2cSJim Jagielski 			currentQuery.push_back( toliterate );
122*b1cdbd2cSJim Jagielski 			queryList.push_back( currentQuery );
123*b1cdbd2cSJim Jagielski 
124*b1cdbd2cSJim Jagielski 			int nCpy = 1 + idx;
125*b1cdbd2cSJim Jagielski 			if( nCpy >= query.getLength() )
126*b1cdbd2cSJim Jagielski 				query = rtl::OUString();
127*b1cdbd2cSJim Jagielski 			else
128*b1cdbd2cSJim Jagielski 				query = query.copy( 1 + idx );
129*b1cdbd2cSJim Jagielski 		}
130*b1cdbd2cSJim Jagielski 	}
131*b1cdbd2cSJim Jagielski 
132*b1cdbd2cSJim Jagielski 	vector< rtl::OUString > aCompleteResultVector;
133*b1cdbd2cSJim Jagielski 	if( xInvocation.is() )
134*b1cdbd2cSJim Jagielski 	{
135*b1cdbd2cSJim Jagielski 		rtl::OUString scope = m_aURLParameter.get_scope();
136*b1cdbd2cSJim Jagielski 		bool bCaptionsOnly = ( scope.compareToAscii( "Heading" ) == 0 );
137*b1cdbd2cSJim Jagielski 		sal_Int32 hitCount = m_aURLParameter.get_hitCount();
138*b1cdbd2cSJim Jagielski 
139*b1cdbd2cSJim Jagielski #ifdef LOGGING
140*b1cdbd2cSJim Jagielski 		FILE* pFile = fopen( "d:\\resultset_out.txt", "w" );
141*b1cdbd2cSJim Jagielski #endif
142*b1cdbd2cSJim Jagielski 
143*b1cdbd2cSJim Jagielski 		IndexFolderIterator aIndexFolderIt( *pDatabases, m_aURLParameter.get_module(), m_aURLParameter.get_language() );
144*b1cdbd2cSJim Jagielski 		rtl::OUString idxDir;
145*b1cdbd2cSJim Jagielski 		bool bExtension = false;
146*b1cdbd2cSJim Jagielski 		int iDir = 0;
147*b1cdbd2cSJim Jagielski 		vector< vector<HitItem>* > aIndexFolderResultVectorVector;
148*b1cdbd2cSJim Jagielski 
149*b1cdbd2cSJim Jagielski 		bool bTemporary;
150*b1cdbd2cSJim Jagielski 		while( (idxDir = aIndexFolderIt.nextIndexFolder( bExtension, bTemporary )).getLength() > 0 )
151*b1cdbd2cSJim Jagielski 		{
152*b1cdbd2cSJim Jagielski 			vector<HitItem> aIndexFolderResultVector;
153*b1cdbd2cSJim Jagielski 
154*b1cdbd2cSJim Jagielski 			try
155*b1cdbd2cSJim Jagielski 			{
156*b1cdbd2cSJim Jagielski 				vector< vector<HitItem>* > aQueryListResultVectorVector;
157*b1cdbd2cSJim Jagielski 				set< rtl::OUString > aSet,aCurrent,aResultSet;
158*b1cdbd2cSJim Jagielski 
159*b1cdbd2cSJim Jagielski 				int nQueryListSize = queryList.size();
160*b1cdbd2cSJim Jagielski 				if( nQueryListSize > 1 )
161*b1cdbd2cSJim Jagielski 					hitCount = 2000;
162*b1cdbd2cSJim Jagielski 
163*b1cdbd2cSJim Jagielski 				for( int i = 0; i < nQueryListSize; ++i )
164*b1cdbd2cSJim Jagielski 				{
165*b1cdbd2cSJim Jagielski 					vector<HitItem>* pQueryResultVector;
166*b1cdbd2cSJim Jagielski 					if( nQueryListSize > 1 )
167*b1cdbd2cSJim Jagielski 					{
168*b1cdbd2cSJim Jagielski 						pQueryResultVector = new vector<HitItem>();
169*b1cdbd2cSJim Jagielski 						aQueryListResultVectorVector.push_back( pQueryResultVector );
170*b1cdbd2cSJim Jagielski 					}
171*b1cdbd2cSJim Jagielski 					else
172*b1cdbd2cSJim Jagielski 					{
173*b1cdbd2cSJim Jagielski 						pQueryResultVector = &aIndexFolderResultVector;
174*b1cdbd2cSJim Jagielski 					}
175*b1cdbd2cSJim Jagielski 					pQueryResultVector->reserve( hitCount );
176*b1cdbd2cSJim Jagielski 
177*b1cdbd2cSJim Jagielski 					int nParamCount = bCaptionsOnly ? 7 : 6;
178*b1cdbd2cSJim Jagielski 					Sequence<uno::Any> aParamsSeq( nParamCount );
179*b1cdbd2cSJim Jagielski 
180*b1cdbd2cSJim Jagielski 					aParamsSeq[0] = uno::makeAny( rtl::OUString::createFromAscii( "-lang" ) );
181*b1cdbd2cSJim Jagielski 					aParamsSeq[1] = uno::makeAny( m_aURLParameter.get_language() );
182*b1cdbd2cSJim Jagielski 
183*b1cdbd2cSJim Jagielski 					aParamsSeq[2] = uno::makeAny( rtl::OUString::createFromAscii( "-index" ) );
184*b1cdbd2cSJim Jagielski 					rtl::OUString aSystemPath;
185*b1cdbd2cSJim Jagielski 					osl::FileBase::getSystemPathFromFileURL( idxDir, aSystemPath );
186*b1cdbd2cSJim Jagielski 					aParamsSeq[3] = uno::makeAny( aSystemPath );
187*b1cdbd2cSJim Jagielski 
188*b1cdbd2cSJim Jagielski 					aParamsSeq[4] = uno::makeAny( rtl::OUString::createFromAscii( "-query" ) );
189*b1cdbd2cSJim Jagielski 
190*b1cdbd2cSJim Jagielski 					const std::vector< rtl::OUString >& aListItem = queryList[i];
191*b1cdbd2cSJim Jagielski 					::rtl::OUString aNewQueryStr = aListItem[0];
192*b1cdbd2cSJim Jagielski 					aParamsSeq[5] = uno::makeAny( aNewQueryStr );
193*b1cdbd2cSJim Jagielski 
194*b1cdbd2cSJim Jagielski 					if( bCaptionsOnly )
195*b1cdbd2cSJim Jagielski 						aParamsSeq[6] = uno::makeAny( rtl::OUString::createFromAscii( "-caption" ) );
196*b1cdbd2cSJim Jagielski 
197*b1cdbd2cSJim Jagielski 					Sequence< sal_Int16 > aOutParamIndex;
198*b1cdbd2cSJim Jagielski 					Sequence< uno::Any > aOutParam;
199*b1cdbd2cSJim Jagielski 
200*b1cdbd2cSJim Jagielski 					uno::Any aRet = xInvocation->invoke( rtl::OUString::createFromAscii( "search" ),
201*b1cdbd2cSJim Jagielski 						aParamsSeq, aOutParamIndex, aOutParam );
202*b1cdbd2cSJim Jagielski 
203*b1cdbd2cSJim Jagielski 					Sequence< float > aScoreSeq;
204*b1cdbd2cSJim Jagielski 					int nScoreCount = 0;
205*b1cdbd2cSJim Jagielski 					int nOutParamCount = aOutParam.getLength();
206*b1cdbd2cSJim Jagielski 					if( nOutParamCount == 1 )
207*b1cdbd2cSJim Jagielski 					{
208*b1cdbd2cSJim Jagielski 						const uno::Any* pScoreAnySeq = aOutParam.getConstArray();
209*b1cdbd2cSJim Jagielski 						if( pScoreAnySeq[0] >>= aScoreSeq )
210*b1cdbd2cSJim Jagielski 							nScoreCount = aScoreSeq.getLength();
211*b1cdbd2cSJim Jagielski 					}
212*b1cdbd2cSJim Jagielski 
213*b1cdbd2cSJim Jagielski 					Sequence<rtl::OUString> aRetSeq;
214*b1cdbd2cSJim Jagielski 					if( aRet >>= aRetSeq )
215*b1cdbd2cSJim Jagielski 					{
216*b1cdbd2cSJim Jagielski 						if( nQueryListSize > 1 )
217*b1cdbd2cSJim Jagielski 							aSet.clear();
218*b1cdbd2cSJim Jagielski 
219*b1cdbd2cSJim Jagielski 						const rtl::OUString* pRetSeq = aRetSeq.getConstArray();
220*b1cdbd2cSJim Jagielski 						int nCount = aRetSeq.getLength();
221*b1cdbd2cSJim Jagielski 						if( nCount > hitCount )
222*b1cdbd2cSJim Jagielski 							nCount = hitCount;
223*b1cdbd2cSJim Jagielski 						for( int j = 0 ; j < nCount ; ++j )
224*b1cdbd2cSJim Jagielski 						{
225*b1cdbd2cSJim Jagielski 							float fScore = 0.0;
226*b1cdbd2cSJim Jagielski 							if( j < nScoreCount )
227*b1cdbd2cSJim Jagielski 								fScore = aScoreSeq[j];
228*b1cdbd2cSJim Jagielski 
229*b1cdbd2cSJim Jagielski 							rtl::OUString aURL = pRetSeq[j];
230*b1cdbd2cSJim Jagielski 							pQueryResultVector->push_back( HitItem( aURL, fScore ) );
231*b1cdbd2cSJim Jagielski 							if( nQueryListSize > 1 )
232*b1cdbd2cSJim Jagielski 								aSet.insert( aURL );
233*b1cdbd2cSJim Jagielski 
234*b1cdbd2cSJim Jagielski #ifdef LOGGING
235*b1cdbd2cSJim Jagielski 							if( pFile )
236*b1cdbd2cSJim Jagielski 							{
237*b1cdbd2cSJim Jagielski 								rtl::OString tmp(rtl::OUStringToOString( aURL, RTL_TEXTENCODING_UTF8));
238*b1cdbd2cSJim Jagielski 								fprintf( pFile, "Dir %d, Query %d, Item: score=%f, URL=%s\n", iDir, i, fScore, tmp.getStr() );
239*b1cdbd2cSJim Jagielski 							}
240*b1cdbd2cSJim Jagielski #endif
241*b1cdbd2cSJim Jagielski 						}
242*b1cdbd2cSJim Jagielski 					}
243*b1cdbd2cSJim Jagielski 
244*b1cdbd2cSJim Jagielski 					// intersect
245*b1cdbd2cSJim Jagielski 					if( nQueryListSize > 1 )
246*b1cdbd2cSJim Jagielski 					{
247*b1cdbd2cSJim Jagielski 						if( i == 0 )
248*b1cdbd2cSJim Jagielski 						{
249*b1cdbd2cSJim Jagielski 							aResultSet = aSet;
250*b1cdbd2cSJim Jagielski 						}
251*b1cdbd2cSJim Jagielski 						else
252*b1cdbd2cSJim Jagielski 						{
253*b1cdbd2cSJim Jagielski 							aCurrent = aResultSet;
254*b1cdbd2cSJim Jagielski 							aResultSet.clear();
255*b1cdbd2cSJim Jagielski 							set_intersection( aSet.begin(),aSet.end(),
256*b1cdbd2cSJim Jagielski 											  aCurrent.begin(),aCurrent.end(),
257*b1cdbd2cSJim Jagielski 											  inserter(aResultSet,aResultSet.begin()));
258*b1cdbd2cSJim Jagielski 						}
259*b1cdbd2cSJim Jagielski 					}
260*b1cdbd2cSJim Jagielski 				}
261*b1cdbd2cSJim Jagielski 
262*b1cdbd2cSJim Jagielski 				// Combine results in aIndexFolderResultVector
263*b1cdbd2cSJim Jagielski 				if( nQueryListSize > 1 )
264*b1cdbd2cSJim Jagielski 				{
265*b1cdbd2cSJim Jagielski 					for( int n = 0 ; n < nQueryListSize ; ++n )
266*b1cdbd2cSJim Jagielski 					{
267*b1cdbd2cSJim Jagielski 						vector<HitItem>* pQueryResultVector = aQueryListResultVectorVector[n];
268*b1cdbd2cSJim Jagielski 						vector<HitItem>& rQueryResultVector = *pQueryResultVector;
269*b1cdbd2cSJim Jagielski 
270*b1cdbd2cSJim Jagielski 						int nItemCount = rQueryResultVector.size();
271*b1cdbd2cSJim Jagielski 						for( int i = 0 ; i < nItemCount ; ++i )
272*b1cdbd2cSJim Jagielski 						{
273*b1cdbd2cSJim Jagielski 							const HitItem& rItem = rQueryResultVector[ i ];
274*b1cdbd2cSJim Jagielski 							set< rtl::OUString >::iterator it;
275*b1cdbd2cSJim Jagielski 							if( (it = aResultSet.find( rItem.m_aURL )) != aResultSet.end() )
276*b1cdbd2cSJim Jagielski 							{
277*b1cdbd2cSJim Jagielski 								HitItem aItemCopy( rItem );
278*b1cdbd2cSJim Jagielski 								aItemCopy.m_fScore /= nQueryListSize;	// To get average score
279*b1cdbd2cSJim Jagielski 								if( n == 0 )
280*b1cdbd2cSJim Jagielski 								{
281*b1cdbd2cSJim Jagielski 									// Use first pass to create entry
282*b1cdbd2cSJim Jagielski 									aIndexFolderResultVector.push_back( aItemCopy );
283*b1cdbd2cSJim Jagielski 
284*b1cdbd2cSJim Jagielski #ifdef LOGGING
285*b1cdbd2cSJim Jagielski 									if( pFile )
286*b1cdbd2cSJim Jagielski 									{
287*b1cdbd2cSJim Jagielski 										rtl::OString tmp(rtl::OUStringToOString( aItemCopy.m_aURL, RTL_TEXTENCODING_UTF8));
288*b1cdbd2cSJim Jagielski 										fprintf( pFile, "Combine: Query %d (first pass), Item %d: score=%f (%f), URL=%s\n", n, i, aItemCopy.m_fScore, rItem.m_fScore, tmp.getStr() );
289*b1cdbd2cSJim Jagielski 									}
290*b1cdbd2cSJim Jagielski #endif
291*b1cdbd2cSJim Jagielski 								}
292*b1cdbd2cSJim Jagielski 								else
293*b1cdbd2cSJim Jagielski 								{
294*b1cdbd2cSJim Jagielski 									// Find entry in vector
295*b1cdbd2cSJim Jagielski 									int nCount = aIndexFolderResultVector.size();
296*b1cdbd2cSJim Jagielski 									for( int j = 0 ; j < nCount ; ++j )
297*b1cdbd2cSJim Jagielski 									{
298*b1cdbd2cSJim Jagielski 										HitItem& rFindItem = aIndexFolderResultVector[ j ];
299*b1cdbd2cSJim Jagielski 										if( rFindItem.m_aURL.equals( aItemCopy.m_aURL ) )
300*b1cdbd2cSJim Jagielski 										{
301*b1cdbd2cSJim Jagielski #ifdef LOGGING
302*b1cdbd2cSJim Jagielski 											if( pFile )
303*b1cdbd2cSJim Jagielski 											{
304*b1cdbd2cSJim Jagielski 												rtl::OString tmp(rtl::OUStringToOString( aItemCopy.m_aURL, RTL_TEXTENCODING_UTF8));
305*b1cdbd2cSJim Jagielski 												fprintf( pFile, "Combine: Query %d, Item %d: score=%f + %f = %f, URL=%s\n", n, i,
306*b1cdbd2cSJim Jagielski 													rFindItem.m_fScore, aItemCopy.m_fScore, rFindItem.m_fScore + aItemCopy.m_fScore, tmp.getStr() );
307*b1cdbd2cSJim Jagielski 											}
308*b1cdbd2cSJim Jagielski #endif
309*b1cdbd2cSJim Jagielski 
310*b1cdbd2cSJim Jagielski 											rFindItem.m_fScore += aItemCopy.m_fScore;
311*b1cdbd2cSJim Jagielski 											break;
312*b1cdbd2cSJim Jagielski 										}
313*b1cdbd2cSJim Jagielski 									}
314*b1cdbd2cSJim Jagielski 								}
315*b1cdbd2cSJim Jagielski 							}
316*b1cdbd2cSJim Jagielski 						}
317*b1cdbd2cSJim Jagielski 
318*b1cdbd2cSJim Jagielski 						delete pQueryResultVector;
319*b1cdbd2cSJim Jagielski 					}
320*b1cdbd2cSJim Jagielski 
321*b1cdbd2cSJim Jagielski 					sort( aIndexFolderResultVector.begin(), aIndexFolderResultVector.end() );
322*b1cdbd2cSJim Jagielski 				}
323*b1cdbd2cSJim Jagielski 
324*b1cdbd2cSJim Jagielski 				vector<HitItem>* pIndexFolderHitItemVector = new vector<HitItem>( aIndexFolderResultVector );
325*b1cdbd2cSJim Jagielski 				aIndexFolderResultVectorVector.push_back( pIndexFolderHitItemVector );
326*b1cdbd2cSJim Jagielski 				aIndexFolderResultVector.clear();
327*b1cdbd2cSJim Jagielski 			}
328*b1cdbd2cSJim Jagielski 			catch( const Exception& )
329*b1cdbd2cSJim Jagielski 			{
330*b1cdbd2cSJim Jagielski 			}
331*b1cdbd2cSJim Jagielski 
332*b1cdbd2cSJim Jagielski 			++iDir;
333*b1cdbd2cSJim Jagielski 
334*b1cdbd2cSJim Jagielski 			if( bTemporary )
335*b1cdbd2cSJim Jagielski 				aIndexFolderIt.deleteTempIndexFolder( idxDir );
336*b1cdbd2cSJim Jagielski 
337*b1cdbd2cSJim Jagielski 		}	// Iterator
338*b1cdbd2cSJim Jagielski 
339*b1cdbd2cSJim Jagielski 
340*b1cdbd2cSJim Jagielski 		int nVectorCount = aIndexFolderResultVectorVector.size();
341*b1cdbd2cSJim Jagielski 		vector<HitItem>::size_type* pCurrentVectorIndex = new vector<HitItem>::size_type[nVectorCount];
342*b1cdbd2cSJim Jagielski 		for( int j = 0 ; j < nVectorCount ; ++j )
343*b1cdbd2cSJim Jagielski 			pCurrentVectorIndex[j] = 0;
344*b1cdbd2cSJim Jagielski 
345*b1cdbd2cSJim Jagielski #ifdef LOGGING
346*b1cdbd2cSJim Jagielski 		if( pFile )
347*b1cdbd2cSJim Jagielski 		{
348*b1cdbd2cSJim Jagielski 			for( int k = 0 ; k < nVectorCount ; ++k )
349*b1cdbd2cSJim Jagielski 			{
350*b1cdbd2cSJim Jagielski 				vector<HitItem>& rIndexFolderVector = *aIndexFolderResultVectorVector[k];
351*b1cdbd2cSJim Jagielski 				int nItemCount = rIndexFolderVector.size();
352*b1cdbd2cSJim Jagielski 
353*b1cdbd2cSJim Jagielski 				fprintf( pFile, "Vector %d, %d elements\n", k, nItemCount );
354*b1cdbd2cSJim Jagielski 
355*b1cdbd2cSJim Jagielski 				for( int i = 0 ; i < nItemCount ; ++i )
356*b1cdbd2cSJim Jagielski 				{
357*b1cdbd2cSJim Jagielski 					const HitItem& rItem = rIndexFolderVector[ i ];
358*b1cdbd2cSJim Jagielski 					rtl::OString tmp(rtl::OUStringToOString(rItem.m_aURL, RTL_TEXTENCODING_UTF8));
359*b1cdbd2cSJim Jagielski 					fprintf( pFile, "    Item_vector%d, %d/%d: score=%f, URL=%s\n", k, i, nItemCount, rItem.m_fScore, tmp.getStr() );
360*b1cdbd2cSJim Jagielski 				}
361*b1cdbd2cSJim Jagielski 			}
362*b1cdbd2cSJim Jagielski 		}
363*b1cdbd2cSJim Jagielski #endif
364*b1cdbd2cSJim Jagielski 
365*b1cdbd2cSJim Jagielski 		sal_Int32 nTotalHitCount = m_aURLParameter.get_hitCount();
366*b1cdbd2cSJim Jagielski 		sal_Int32 nHitCount = 0;
367*b1cdbd2cSJim Jagielski 		while( nHitCount < nTotalHitCount )
368*b1cdbd2cSJim Jagielski 		{
369*b1cdbd2cSJim Jagielski 			int iVectorWithBestScore = -1;
370*b1cdbd2cSJim Jagielski 			float fBestScore = 0.0;
371*b1cdbd2cSJim Jagielski 			for( int k = 0 ; k < nVectorCount ; ++k )
372*b1cdbd2cSJim Jagielski 			{
373*b1cdbd2cSJim Jagielski 				vector<HitItem>& rIndexFolderVector = *aIndexFolderResultVectorVector[k];
374*b1cdbd2cSJim Jagielski 				if( pCurrentVectorIndex[k] < rIndexFolderVector.size() )
375*b1cdbd2cSJim Jagielski 				{
376*b1cdbd2cSJim Jagielski 					const HitItem& rItem = rIndexFolderVector[ pCurrentVectorIndex[k] ];
377*b1cdbd2cSJim Jagielski 
378*b1cdbd2cSJim Jagielski 					if( fBestScore < rItem.m_fScore )
379*b1cdbd2cSJim Jagielski 					{
380*b1cdbd2cSJim Jagielski 						fBestScore = rItem.m_fScore;
381*b1cdbd2cSJim Jagielski 						iVectorWithBestScore = k;
382*b1cdbd2cSJim Jagielski 					}
383*b1cdbd2cSJim Jagielski 				}
384*b1cdbd2cSJim Jagielski 			}
385*b1cdbd2cSJim Jagielski 
386*b1cdbd2cSJim Jagielski 			if( iVectorWithBestScore == -1 )	// No item left at all
387*b1cdbd2cSJim Jagielski 				break;
388*b1cdbd2cSJim Jagielski 
389*b1cdbd2cSJim Jagielski 			vector<HitItem>& rIndexFolderVector = *aIndexFolderResultVectorVector[iVectorWithBestScore];
390*b1cdbd2cSJim Jagielski 			const HitItem& rItem = rIndexFolderVector[ pCurrentVectorIndex[iVectorWithBestScore] ];
391*b1cdbd2cSJim Jagielski 
392*b1cdbd2cSJim Jagielski 			pCurrentVectorIndex[iVectorWithBestScore]++;
393*b1cdbd2cSJim Jagielski 
394*b1cdbd2cSJim Jagielski 			aCompleteResultVector.push_back( rItem.m_aURL );
395*b1cdbd2cSJim Jagielski 			++nHitCount;
396*b1cdbd2cSJim Jagielski 		}
397*b1cdbd2cSJim Jagielski 
398*b1cdbd2cSJim Jagielski 		delete[] pCurrentVectorIndex;
399*b1cdbd2cSJim Jagielski 		for( int n = 0 ; n < nVectorCount ; ++n )
400*b1cdbd2cSJim Jagielski 		{
401*b1cdbd2cSJim Jagielski 			vector<HitItem>* pIndexFolderVector = aIndexFolderResultVectorVector[n];
402*b1cdbd2cSJim Jagielski 			delete pIndexFolderVector;
403*b1cdbd2cSJim Jagielski 		}
404*b1cdbd2cSJim Jagielski 
405*b1cdbd2cSJim Jagielski #ifdef LOGGING
406*b1cdbd2cSJim Jagielski 		fclose( pFile );
407*b1cdbd2cSJim Jagielski #endif
408*b1cdbd2cSJim Jagielski 	}
409*b1cdbd2cSJim Jagielski 
410*b1cdbd2cSJim Jagielski 	sal_Int32 replIdx = rtl::OUString::createFromAscii( "#HLP#" ).getLength();
411*b1cdbd2cSJim Jagielski 	rtl::OUString replWith = rtl::OUString::createFromAscii( "vnd.sun.star.help://" );
412*b1cdbd2cSJim Jagielski 
413*b1cdbd2cSJim Jagielski 	int nResultCount = aCompleteResultVector.size();
414*b1cdbd2cSJim Jagielski 	for( int r = 0 ; r < nResultCount ; ++r )
415*b1cdbd2cSJim Jagielski 	{
416*b1cdbd2cSJim Jagielski 		rtl::OUString aURL = aCompleteResultVector[r];
417*b1cdbd2cSJim Jagielski 		rtl::OUString aResultStr = replWith + aURL.copy(replIdx);
418*b1cdbd2cSJim Jagielski   		m_aPath.push_back( aResultStr );
419*b1cdbd2cSJim Jagielski 	}
420*b1cdbd2cSJim Jagielski 
421*b1cdbd2cSJim Jagielski 	m_aItems.resize( m_aPath.size() );
422*b1cdbd2cSJim Jagielski 	m_aIdents.resize( m_aPath.size() );
423*b1cdbd2cSJim Jagielski 
424*b1cdbd2cSJim Jagielski 	Command aCommand;
425*b1cdbd2cSJim Jagielski 	aCommand.Name = rtl::OUString::createFromAscii( "getPropertyValues" );
426*b1cdbd2cSJim Jagielski 	aCommand.Argument <<= m_sProperty;
427*b1cdbd2cSJim Jagielski 
428*b1cdbd2cSJim Jagielski 	for( m_nRow = 0; sal::static_int_cast<sal_uInt32>( m_nRow ) < m_aPath.size(); ++m_nRow )
429*b1cdbd2cSJim Jagielski 	{
430*b1cdbd2cSJim Jagielski 		m_aPath[m_nRow] =
431*b1cdbd2cSJim Jagielski 			m_aPath[m_nRow]                                          +
432*b1cdbd2cSJim Jagielski 			rtl::OUString::createFromAscii( "?Language=" )           +
433*b1cdbd2cSJim Jagielski 			m_aURLParameter.get_language()                           +
434*b1cdbd2cSJim Jagielski 			rtl::OUString::createFromAscii( "&System=" )             +
435*b1cdbd2cSJim Jagielski 			m_aURLParameter.get_system();
436*b1cdbd2cSJim Jagielski 
437*b1cdbd2cSJim Jagielski 		uno::Reference< XContent > content = queryContent();
438*b1cdbd2cSJim Jagielski 		if( content.is() )
439*b1cdbd2cSJim Jagielski 		{
440*b1cdbd2cSJim Jagielski 			uno::Reference< XCommandProcessor > cmd( content,uno::UNO_QUERY );
441*b1cdbd2cSJim Jagielski 			cmd->execute( aCommand,0,uno::Reference< XCommandEnvironment >( 0 ) ) >>= m_aItems[m_nRow]; //TODO: check return value of operator >>=
442*b1cdbd2cSJim Jagielski 		}
443*b1cdbd2cSJim Jagielski 	}
444*b1cdbd2cSJim Jagielski 	m_nRow = 0xffffffff;
445*b1cdbd2cSJim Jagielski }
446