xref: /aoo42x/main/sd/source/ui/unoidl/unosrch.cxx (revision 5b190011)
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_sd.hxx"
26 #include <vcl/svapp.hxx>
27 #include <vos/mutex.hxx>
28 
29 #include <svx/unoshape.hxx>
30 #include <svx/svdpool.hxx>
31 #include <svx/unoprov.hxx>
32 #include <editeng/unotext.hxx>
33 
34 #include <comphelper/extract.hxx>
35 #include <rtl/uuid.h>
36 #include <rtl/memory.h>
37 
38 #include "unohelp.hxx"
39 #include "unoprnms.hxx"
40 #include "unosrch.hxx"
41 
42 using namespace ::vos;
43 using namespace ::rtl;
44 using namespace ::com::sun::star;
45 
46 #define WID_SEARCH_BACKWARDS	0
47 #define WID_SEARCH_CASE			1
48 #define WID_SEARCH_WORDS		2
49 
50 const SfxItemPropertyMapEntry* ImplGetSearchPropertyMap()
51 {
52 	static const SfxItemPropertyMapEntry aSearchPropertyMap_Impl[] =
53 	{
54 		{ MAP_CHAR_LEN(UNO_NAME_SEARCH_BACKWARDS),	WID_SEARCH_BACKWARDS,	&::getBooleanCppuType(),	0,	0 },
55 		{ MAP_CHAR_LEN(UNO_NAME_SEARCH_CASE),		WID_SEARCH_CASE,		&::getBooleanCppuType(),	0,	0 },
56 		{ MAP_CHAR_LEN(UNO_NAME_SEARCH_WORDS),		WID_SEARCH_WORDS,		&::getBooleanCppuType(),	0,	0 },
57 		{ 0,0,0,0,0,0}
58 	};
59 
60 	return aSearchPropertyMap_Impl;
61 }
62 
63 class SearchContext_impl
64 {
65 	uno::Reference< drawing::XShapes > mxShapes;
66 	sal_Int32 mnIndex;
67 	SearchContext_impl*	mpParent;
68 
69 public:
70 	SearchContext_impl( uno::Reference< drawing::XShapes >  xShapes, SearchContext_impl* pParent = NULL )
71 		: mxShapes( xShapes ), mnIndex( -1 ), mpParent( pParent ) {}
72 
73 
74 	uno::Reference< drawing::XShape > firstShape()
75 	{
76 		mnIndex = -1;
77 		return nextShape();
78 	}
79 
80 	uno::Reference< drawing::XShape > nextShape()
81 	{
82 		uno::Reference< drawing::XShape >  xShape;
83 		mnIndex++;
84 		if( mxShapes.is() && mxShapes->getCount() > mnIndex )
85 		{
86 			mxShapes->getByIndex( mnIndex ) >>= xShape;
87 		}
88 		return xShape;
89 	}
90 
91 	SearchContext_impl* getParent() const { return mpParent; }
92 };
93 
94 /* ================================================================= */
95 /** this class implements a search or replace operation on a given
96 	page or a given sdrobj
97   */
98 
99 SdUnoSearchReplaceShape::SdUnoSearchReplaceShape( drawing::XDrawPage* pPage ) throw()
100 {
101 	mpPage = pPage;
102 }
103 
104 SdUnoSearchReplaceShape::~SdUnoSearchReplaceShape() throw()
105 {
106 }
107 
108 // util::XReplaceable
109 uno::Reference< util::XReplaceDescriptor > SAL_CALL SdUnoSearchReplaceShape::createReplaceDescriptor()
110 	throw( uno::RuntimeException )
111 {
112 	return new SdUnoSearchReplaceDescriptor(sal_True);
113 }
114 
115 sal_Int32 SAL_CALL SdUnoSearchReplaceShape::replaceAll( const uno::Reference< util::XSearchDescriptor >& xDesc )
116 	throw( uno::RuntimeException )
117 {
118 	SdUnoSearchReplaceDescriptor* pDescr = SdUnoSearchReplaceDescriptor::getImplementation( xDesc );
119 	if( pDescr == NULL )
120 		return 0;
121 
122 	sal_Int32 nFound	= 0;
123 
124 	uno::Reference< drawing::XShapes >  xShapes;
125 	uno::Reference< drawing::XShape >  xShape;
126 
127 	SearchContext_impl* pContext = NULL;
128 	if(mpPage)
129 	{
130 		uno::Reference< drawing::XDrawPage > xPage( mpPage );
131 
132 		xPage->queryInterface( ITYPE( drawing::XShapes ) ) >>= xShapes;
133 
134 		if( xShapes.is() && (xShapes->getCount() > 0) )
135 		{
136 			pContext = new SearchContext_impl( xShapes );
137 			xShape = pContext->firstShape();
138 		}
139 		else
140 		{
141 			xShapes = NULL;
142 		}
143 	}
144 	else
145 	{
146 		xShape = mpShape;
147 	}
148 
149 	while( xShape.is() )
150 	{
151 		// replace in xShape
152 		uno::Reference< text::XText >  xText(xShape, uno::UNO_QUERY);
153 		uno::Reference< text::XTextRange >  xRange(xText, uno::UNO_QUERY);
154 		uno::Reference< text::XTextRange >  xFound;
155 
156 		while( xRange.is() )
157 		{
158 			xFound = Search( xRange, pDescr );
159 			if( !xFound.is() )
160 				break;
161 
162 			xFound->setString( pDescr->getReplaceString() );
163 			xRange = xFound->getEnd();
164 			nFound++;
165 		}
166 		// done with xShape -> get next shape
167 
168 		// test if its a group
169 		uno::Reference< drawing::XShapes > xGroupShape( xShape, uno::UNO_QUERY );
170 		if( xGroupShape.is() && ( xGroupShape->getCount() > 0 ) )
171 		{
172 			pContext = new SearchContext_impl( xGroupShape, pContext );
173 			xShape = pContext->firstShape();
174 		}
175 		else
176 		{
177 			if( pContext )
178 				xShape = pContext->nextShape();
179 			else
180 				xShape = NULL;
181 		}
182 
183 		// test parent contexts for next shape if none
184 		// is found in the current context
185 		while( pContext && !xShape.is() )
186 		{
187 			if( pContext->getParent() )
188 			{
189 				SearchContext_impl* pOldContext = pContext;
190 				pContext = pContext->getParent();
191 				delete pOldContext;
192 				xShape = pContext->nextShape();
193 			}
194 			else
195 			{
196 				delete pContext;
197 				pContext = NULL;
198 				xShape = NULL;
199 			}
200 		}
201 	}
202 
203 	return nFound;
204 }
205 
206 // XSearchable
207 uno::Reference< ::com::sun::star::util::XSearchDescriptor > SAL_CALL SdUnoSearchReplaceShape::createSearchDescriptor(  )
208 	throw(::com::sun::star::uno::RuntimeException)
209 {
210 	return new SdUnoSearchReplaceDescriptor(sal_False);
211 }
212 
213 uno::Reference< ::com::sun::star::container::XIndexAccess > SAL_CALL SdUnoSearchReplaceShape::findAll( const ::com::sun::star::uno::Reference< ::com::sun::star::util::XSearchDescriptor >& xDesc )
214 	throw(::com::sun::star::uno::RuntimeException)
215 {
216 	SdUnoSearchReplaceDescriptor* pDescr = SdUnoSearchReplaceDescriptor::getImplementation( xDesc );
217 	if( pDescr == NULL )
218 		return uno::Reference< container::XIndexAccess > ();
219 
220 
221 	sal_Int32 nSequence = 32;
222 	sal_Int32 nFound    = 0;
223 
224 	uno::Sequence < uno::Reference< uno::XInterface >  > aSeq( nSequence );
225 
226 	uno::Reference< uno::XInterface > * pArray = aSeq.getArray();
227 
228 	uno::Reference< drawing::XShapes >  xShapes;
229 	uno::Reference< drawing::XShape >  xShape;
230 
231 	SearchContext_impl* pContext = NULL;
232 	if(mpPage)
233 	{
234 		uno::Reference< drawing::XDrawPage >  xPage( mpPage );
235 		xPage->queryInterface( ITYPE( drawing::XShapes ) ) >>= xShapes;
236 
237 		if( xShapes.is() && xShapes->getCount() > 0 )
238 		{
239 			pContext = new SearchContext_impl( xShapes );
240 			xShape = pContext->firstShape();
241 		}
242 		else
243 		{
244 			xShapes = NULL;
245 		}
246 	}
247 	else
248 	{
249 		xShape = mpShape;
250 	}
251 	while( xShape.is() )
252 	{
253 		// find in xShape
254 		uno::Reference< text::XText >  xText(xShape, uno::UNO_QUERY);
255 		uno::Reference< text::XTextRange >  xRange(xText, uno::UNO_QUERY);
256 		uno::Reference< text::XTextRange >  xFound;
257 
258 		while( xRange.is() )
259 		{
260 			xFound = Search( xRange, pDescr );
261 			if( !xFound.is() )
262 				break;
263 
264 			if( nFound >= nSequence )
265 			{
266 				nSequence += 32;
267 				aSeq.realloc( nSequence );
268 				pArray = aSeq.getArray();
269 			}
270 
271 			pArray[nFound++] = xFound;
272 
273 			xRange = xFound->getEnd();
274 		}
275 		// done with shape -> get next shape
276 
277 		// test if its a group
278 		uno::Reference< drawing::XShapes >  xGroupShape;
279 		uno::Any aAny( xShape->queryInterface( ITYPE( drawing::XShapes )));
280 
281 		if( (aAny >>= xGroupShape ) && xGroupShape->getCount() > 0 )
282 		{
283 			pContext = new SearchContext_impl( xGroupShape, pContext );
284 			xShape = pContext->firstShape();
285 		}
286 		else
287 		{
288 			if( pContext )
289 				xShape = pContext->nextShape();
290 			else
291 				xShape = NULL;
292 		}
293 
294 		// test parent contexts for next shape if none
295 		// is found in the current context
296 		while( pContext && !xShape.is() )
297 		{
298 			if( pContext->getParent() )
299 			{
300 				SearchContext_impl* pOldContext = pContext;
301 				pContext = pContext->getParent();
302 				delete pOldContext;
303 				xShape = pContext->nextShape();
304 			}
305 			else
306 			{
307 				delete pContext;
308 				pContext = NULL;
309 				xShape = NULL;
310 			}
311 		}
312 	}
313 
314 	if( nFound != nSequence )
315 		aSeq.realloc( nFound );
316 
317 	return (container::XIndexAccess*)new SdUnoFindAllAccess( aSeq );
318 }
319 
320 uno::Reference< ::com::sun::star::uno::XInterface > SAL_CALL SdUnoSearchReplaceShape::findFirst( const ::com::sun::star::uno::Reference< ::com::sun::star::util::XSearchDescriptor >& xDesc )
321 	throw(::com::sun::star::uno::RuntimeException)
322 {
323 	uno::Reference< text::XTextRange > xRange( GetCurrentShape(), uno::UNO_QUERY );
324 	if( xRange.is() )
325 		return findNext( xRange, xDesc );
326 
327 	return uno::Reference< uno::XInterface > ();
328 }
329 
330 uno::Reference< drawing::XShape >  SdUnoSearchReplaceShape::GetCurrentShape() const throw()
331 {
332 	uno::Reference< drawing::XShape >  xShape;
333 
334 	if( mpPage )
335 	{
336 		uno::Reference< drawing::XDrawPage >  xPage( mpPage );
337 		uno::Reference< container::XIndexAccess >  xShapes( xPage, uno::UNO_QUERY );
338 		if( xShapes.is() )
339 		{
340 			if(xShapes->getCount() > 0)
341 			{
342 				xShapes->getByIndex(0) >>= xShape;
343 			}
344 		}
345 	}
346 	else if( mpShape )
347 	{
348 		xShape = mpShape;
349 	}
350 
351 	return xShape;
352 
353 }
354 
355 uno::Reference< ::com::sun::star::uno::XInterface > SAL_CALL SdUnoSearchReplaceShape::findNext( const ::com::sun::star::uno::Reference< ::com::sun::star::uno::XInterface >& xStartAt, const ::com::sun::star::uno::Reference< ::com::sun::star::util::XSearchDescriptor >& xDesc )
356 	throw(::com::sun::star::uno::RuntimeException)
357 {
358 	SdUnoSearchReplaceDescriptor* pDescr = SdUnoSearchReplaceDescriptor::getImplementation( xDesc );
359 
360 	uno::Reference< uno::XInterface > xFound;
361 
362 	uno::Reference< text::XTextRange > xRange( xStartAt, uno::UNO_QUERY );
363 	if(pDescr && xRange.is() )
364 	{
365 
366 		uno::Reference< text::XTextRange > xCurrentRange( xStartAt, uno::UNO_QUERY );
367 
368 		uno::Reference< drawing::XShape > xCurrentShape( GetShape( xCurrentRange ) );
369 
370 		while(!xFound.is() && xRange.is())
371 		{
372 			xFound = Search( xRange, pDescr );
373 			if(!xFound.is())
374 			{
375 				// we need a new starting range now
376 				xRange = NULL;
377 
378 				if(mpPage)
379 				{
380 					uno::Reference< drawing::XDrawPage >  xPage( mpPage );
381 
382 					// we do a page wide search, so skip to the next shape here
383 					uno::Reference< container::XIndexAccess > xShapes( xPage, uno::UNO_QUERY );
384 
385 					// get next shape on our page
386 					if( xShapes.is() )
387 					{
388 						uno::Reference< drawing::XShape > xFound2( GetNextShape( xShapes, xCurrentShape ) );
389 						if( xFound2.is() && (xFound2.get() != xCurrentShape.get()) )
390 							xCurrentShape = xFound2;
391 						else
392 							xCurrentShape = NULL;
393 
394 						xCurrentShape->queryInterface( ITYPE( text::XTextRange ) ) >>= xRange;
395 						if(!(xCurrentShape.is() && (xRange.is())))
396 							xRange = NULL;
397 					}
398 				}
399 				else
400 				{
401 					// we search only in this shape, so end search if we have
402 					// not found anything
403 				}
404 			}
405 		}
406 	}
407 	return xFound;
408 }
409 
410 /** this method returns the shape that follows xCurrentShape in the shape collection xShapes.
411 	It steps recursive into groupshapes and returns the xCurrentShape if it is the last
412 	shape in this collection */
413 uno::Reference< drawing::XShape >  SdUnoSearchReplaceShape::GetNextShape( uno::Reference< container::XIndexAccess >  xShapes, uno::Reference< drawing::XShape >  xCurrentShape ) throw()
414 {
415 	uno::Reference< drawing::XShape >  xFound;
416 	uno::Any aAny;
417 
418 	if(xShapes.is() && xCurrentShape.is())
419 	{
420 		const sal_Int32 nCount = xShapes->getCount();
421 		for( sal_Int32 i = 0; i < nCount; i++ )
422 		{
423 			uno::Reference< drawing::XShape > xSearchShape;
424 			xShapes->getByIndex(i) >>= xSearchShape;
425 
426 			if( xSearchShape.is() )
427 			{
428 				uno::Reference< container::XIndexAccess > xGroup( xSearchShape, uno::UNO_QUERY );
429 
430 				if( xCurrentShape.get() == xSearchShape.get() )
431 				{
432 					if( xGroup.is() && xGroup->getCount() > 0 )
433 					{
434 						xGroup->getByIndex( 0 ) >>= xFound;
435 					}
436 					else
437 					{
438 						i++;
439 						if( i < nCount )
440 							xShapes->getByIndex( i ) >>= xFound;
441 						else
442 							xFound = xCurrentShape;
443 					}
444 
445 					break;
446 				}
447 				else if( xGroup.is() )
448 				{
449 					xFound = GetNextShape( xGroup, xCurrentShape );
450 					if( xFound.is() )
451 					{
452 						if( xFound.get() == xCurrentShape.get() )
453 						{
454 							// the current shape was found at the end of the group
455 							i++;
456 							if( i < nCount )
457 							{
458 								xShapes->getByIndex(i) >>= xFound;
459 							}
460 						}
461 						break;
462 					}
463 				}
464 			}
465 		}
466 	}
467 
468 	return xFound;
469 }
470 
471 uno::Reference< text::XTextRange >  SdUnoSearchReplaceShape::Search( uno::Reference< text::XTextRange >  xText, SdUnoSearchReplaceDescriptor* pDescr ) throw()
472 {
473 	if(!xText.is())
474 		return uno::Reference< text::XTextRange > ();
475 
476 	uno::Reference< text::XText > xParent( xText->getText() );
477 
478 	if( !xParent.is() )
479 	{
480 		uno::Any aAny( xText->queryInterface( ITYPE( text::XText )) );
481 		aAny >>= xParent;
482 	}
483 
484 	const OUString aText( xParent->getString() );
485 
486 	const sal_Int32 nTextLen = aText.getLength();
487 
488 	sal_Int32* pConvertPos = new sal_Int32[nTextLen+2];
489 	sal_Int32* pConvertPara = new sal_Int32[nTextLen+2];
490 
491 	int ndbg = 0;
492 	const sal_Unicode* pText = aText;
493 
494 	sal_Int32* pPos = pConvertPos;
495 	sal_Int32* pPara = pConvertPara;
496 
497 	sal_Int32 nLastPos = 0, nLastPara = 0;
498 
499 	uno::Reference< container::XEnumerationAccess > xEnumAccess( xParent, uno::UNO_QUERY );
500 
501 	// first we fill the arrys with the position and paragraph for every character
502 	// inside the text
503 	if( xEnumAccess.is() )
504 	{
505 		uno::Reference< container::XEnumeration >  xParaEnum( xEnumAccess->createEnumeration() );
506 
507 		while(xParaEnum->hasMoreElements())
508 		{
509 			uno::Reference< text::XTextContent >  xParagraph( xParaEnum->nextElement(), uno::UNO_QUERY );
510 			if( xParagraph.is() )
511 				xEnumAccess.query( xParagraph );
512 			else
513 				xEnumAccess.clear();
514 
515 			if( xEnumAccess.is() )
516 			{
517 				 uno::Reference< container::XEnumeration >  xPortionEnum( xEnumAccess->createEnumeration() );
518 				 if( xPortionEnum.is() )
519 				 {
520 					while(xPortionEnum->hasMoreElements())
521 					{
522 						uno::Reference< text::XTextRange >  xPortion( xPortionEnum->nextElement(), uno::UNO_QUERY );
523 						if( xPortion.is() )
524 						{
525 							const OUString aPortion( xPortion->getString() );
526 							const sal_Int32 nLen = aPortion.getLength();
527 
528 							ESelection aStartSel( GetSelection( xPortion->getStart() ) );
529 							ESelection aEndSel( GetSelection( xPortion->getEnd() ) );
530 
531 							// special case for empty portions with content or length one portions with content (fields)
532 							if( (aStartSel.nStartPos == aEndSel.nStartPos) || ( (aStartSel.nStartPos == (aEndSel.nStartPos - 1)) && (nLen > 1) ) )
533 							{
534 								for( sal_Int32 i = 0; i < nLen; i++ )
535 								{
536 									if( ndbg < (nTextLen+2) )
537 									{
538 										*pPos++ = aStartSel.nStartPos;
539 										*pPara++ = aStartSel.nStartPara;
540 
541 										ndbg += 1;
542 										pText++;
543 									}
544 									else
545 									{
546 										DBG_ERROR( "array overflow while searching" );
547 									}
548 								}
549 
550 								nLastPos = aStartSel.nStartPos;
551 							}
552 							// normal case
553 							else
554 							{
555 								for( sal_Int32 i = 0; i < nLen; i++ )
556 								{
557 									if( ndbg < (nTextLen+2) )
558 									{
559 										*pPos++ = aStartSel.nStartPos++;
560 										*pPara++ = aStartSel.nStartPara;
561 
562 										ndbg += 1;
563 										pText++;
564 									}
565 									else
566 									{
567 										DBG_ERROR( "array overflow while searching" );
568 									}
569 								}
570 
571 								nLastPos = aStartSel.nStartPos - 1;
572 								DBG_ASSERT( aEndSel.nStartPos == aStartSel.nStartPos, "Search is not working" );
573 							}
574 							nLastPara = aStartSel.nStartPara;
575 						}
576 					}
577 				}
578 			}
579 
580 			if( ndbg < (nTextLen+2) )
581 			{
582 				*pPos++ = nLastPos + 1;
583 				*pPara++ = nLastPara;
584 
585 				ndbg += 1;
586 				pText++;
587 			}
588 			else
589 			{
590 				DBG_ERROR( "array overflow while searching" );
591 			}
592 		}
593 	}
594 
595 	uno::Reference< text::XText >  xFound;
596 	ESelection aSel;
597 
598 	uno::Reference< text::XTextRange > xRangeRef( xText, uno::UNO_QUERY );
599 	if( xRangeRef.is() )
600 		aSel = GetSelection( xRangeRef );
601 
602 	sal_Int32 nStartPos;
603 	sal_Int32 nEndPos   = 0;
604 	for( nStartPos = 0; nStartPos < nTextLen; nStartPos++ )
605 	{
606 		if( pConvertPara[nStartPos] == aSel.nStartPara && pConvertPos[nStartPos] == aSel.nStartPos )
607 			break;
608 	}
609 
610 	if( Search( aText, nStartPos, nEndPos, pDescr ) )
611 	{
612 		if( nStartPos <= nTextLen && nEndPos <= nTextLen )
613 		{
614 			ESelection aSelection( (sal_uInt16)pConvertPara[nStartPos], (sal_uInt16)pConvertPos[nStartPos],
615 							 (sal_uInt16)pConvertPara[nEndPos], (sal_uInt16)pConvertPos[nEndPos] );
616 			SvxUnoTextRange *pRange;
617 
618 			SvxUnoTextBase* pParent = SvxUnoTextBase::getImplementation( xParent );
619 
620 			if(pParent)
621 			{
622 				pRange = new SvxUnoTextRange( *pParent );
623 				xFound = (text::XText*)pRange;
624 				pRange->SetSelection(aSelection);
625 
626 //				pDescr->SetStartPos( nEndPos );
627 			}
628 		}
629 		else
630 		{
631 			DBG_ERROR("Array overflow while searching!");
632 		}
633 	}
634 
635 	delete[] pConvertPos;
636 	delete[] pConvertPara;
637 
638 	return uno::Reference< text::XTextRange > ( xFound, uno::UNO_QUERY );
639 }
640 
641 sal_Bool SdUnoSearchReplaceShape::Search( const OUString& rText, sal_Int32& nStartPos, sal_Int32& nEndPos, SdUnoSearchReplaceDescriptor* pDescr ) throw()
642 {
643 	OUString aSearchStr( pDescr->getSearchString() );
644 	OUString aText( rText );
645 
646 	if( !pDescr->IsCaseSensitive() )
647 	{
648 		aText.toAsciiLowerCase();
649 		aSearchStr.toAsciiLowerCase();
650 	}
651 
652 	sal_Int32 nFound = aText.indexOf( aSearchStr, nStartPos );
653 	if( nFound != -1 )
654 	{
655 		nStartPos = nFound;
656 		nEndPos   = nFound + aSearchStr.getLength();
657 
658 		if(pDescr->IsWords())
659 		{
660 			if( (nStartPos > 0 && aText.getStr()[nStartPos-1] > ' ') ||
661 				(nEndPos < aText.getLength() && aText.getStr()[nEndPos] > ' ') )
662 			{
663 				nStartPos++;
664 				return Search( aText, nStartPos, nEndPos, pDescr );
665 			}
666 		}
667 
668 		return sal_True;
669 	}
670 	else
671 		return sal_False;
672 }
673 
674 ESelection SdUnoSearchReplaceShape::GetSelection( uno::Reference< text::XTextRange >  xTextRange ) throw()
675 {
676 	ESelection aSel;
677 	SvxUnoTextRangeBase* pRange = SvxUnoTextRangeBase::getImplementation( xTextRange );
678 
679 	if(pRange)
680 		aSel = pRange->GetSelection();
681 
682 	return aSel;
683 }
684 
685 uno::Reference< drawing::XShape >  SdUnoSearchReplaceShape::GetShape( uno::Reference< text::XTextRange >  xTextRange ) throw()
686 {
687 	uno::Reference< drawing::XShape >  xShape;
688 
689 	if(xTextRange.is())
690 	{
691 		uno::Reference< text::XText >  xText( xTextRange->getText() );
692 
693 		if(xText.is())
694 		{
695 			do
696 			{
697 				xText->queryInterface( ITYPE( drawing::XShape )) >>= xShape;
698 				if(!xShape.is())
699 				{
700 					uno::Reference< text::XText > xParent( xText->getText() );
701 					if(!xParent.is() || xText.get() == xParent.get())
702 						return xShape;
703 
704 					xText = xParent;
705 				}
706 			} while( !xShape.is() );
707 		}
708 	}
709 
710 	return xShape;
711 }
712 
713 /* ================================================================= */
714 /** this class holds the parameters and status of a search or replace
715 	operation performed by class SdUnoSearchReplaceShape
716   */
717 
718 UNO3_GETIMPLEMENTATION_IMPL( SdUnoSearchReplaceDescriptor );
719 
720 SdUnoSearchReplaceDescriptor::SdUnoSearchReplaceDescriptor( sal_Bool bReplace ) throw()
721 {
722 	mpPropSet = new SvxItemPropertySet(ImplGetSearchPropertyMap(), SdrObject::GetGlobalDrawObjectItemPool());
723 
724 	mbBackwards = sal_False;
725 	mbCaseSensitive = sal_False;
726 	mbWords = sal_False;
727 
728 	mbReplace = bReplace;
729 }
730 
731 SdUnoSearchReplaceDescriptor::~SdUnoSearchReplaceDescriptor() throw()
732 {
733 	delete mpPropSet;
734 }
735 
736 // XSearchDescriptor
737 OUString SAL_CALL SdUnoSearchReplaceDescriptor::getSearchString()
738 	throw(::com::sun::star::uno::RuntimeException)
739 {
740 	return maSearchStr;
741 }
742 
743 void SAL_CALL SdUnoSearchReplaceDescriptor::setSearchString( const OUString& aString )
744 	throw(::com::sun::star::uno::RuntimeException)
745 {
746 	maSearchStr = aString;
747 }
748 
749 // XReplaceDescriptor
750 OUString SAL_CALL SdUnoSearchReplaceDescriptor::getReplaceString()
751 	throw(::com::sun::star::uno::RuntimeException)
752 {
753 	return maReplaceStr;
754 }
755 
756 void SAL_CALL SdUnoSearchReplaceDescriptor::setReplaceString( const ::rtl::OUString& aReplaceString )
757 	throw(::com::sun::star::uno::RuntimeException)
758 {
759 	maReplaceStr = aReplaceString;
760 }
761 
762 // XPropertySet
763 uno::Reference< ::com::sun::star::beans::XPropertySetInfo > SAL_CALL SdUnoSearchReplaceDescriptor::getPropertySetInfo()
764 	throw(::com::sun::star::uno::RuntimeException)
765 {
766 	OGuard aGuard( Application::GetSolarMutex() );
767 	return mpPropSet->getPropertySetInfo();
768 }
769 
770 void SAL_CALL SdUnoSearchReplaceDescriptor::setPropertyValue( const ::rtl::OUString& aPropertyName, const ::com::sun::star::uno::Any& aValue )
771 	throw(::com::sun::star::beans::UnknownPropertyException, ::com::sun::star::beans::PropertyVetoException, ::com::sun::star::lang::IllegalArgumentException, ::com::sun::star::lang::WrappedTargetException, ::com::sun::star::uno::RuntimeException)
772 {
773 	OGuard aGuard( Application::GetSolarMutex() );
774 
775 	const SfxItemPropertySimpleEntry* pEntry = mpPropSet->getPropertyMapEntry(aPropertyName);
776 
777 	sal_Bool bOk = sal_False;
778 
779 	switch( pEntry ? pEntry->nWID : -1 )
780 	{
781 	case WID_SEARCH_BACKWARDS:
782 		bOk = (aValue >>= mbBackwards);
783 		break;
784 	case WID_SEARCH_CASE:
785 		bOk = (aValue >>= mbCaseSensitive);
786 		break;
787 	case WID_SEARCH_WORDS:
788 		bOk = (aValue >>= mbWords);
789 		break;
790 	default:
791 		throw beans::UnknownPropertyException();
792 	}
793 
794 	if( !bOk )
795 		throw lang::IllegalArgumentException();
796 }
797 
798 uno::Any SAL_CALL SdUnoSearchReplaceDescriptor::getPropertyValue( const ::rtl::OUString& PropertyName )
799 	throw(::com::sun::star::beans::UnknownPropertyException, ::com::sun::star::lang::WrappedTargetException, ::com::sun::star::uno::RuntimeException)
800 {
801 	OGuard aGuard( Application::GetSolarMutex() );
802 
803 	uno::Any aAny;
804 
805 	const SfxItemPropertySimpleEntry* pEntry = mpPropSet->getPropertyMapEntry(PropertyName);
806 
807 	switch( pEntry ? pEntry->nWID : -1 )
808 	{
809 	case WID_SEARCH_BACKWARDS:
810 		aAny <<= (sal_Bool)mbBackwards;
811 		break;
812 	case WID_SEARCH_CASE:
813 		aAny <<= (sal_Bool)mbCaseSensitive;
814 		break;
815 	case WID_SEARCH_WORDS:
816 		aAny <<= (sal_Bool)mbWords;
817 		break;
818 	default:
819 		throw beans::UnknownPropertyException();
820 	}
821 
822 	return aAny;
823 }
824 
825 void SAL_CALL SdUnoSearchReplaceDescriptor::addPropertyChangeListener( const ::rtl::OUString& , const ::com::sun::star::uno::Reference< ::com::sun::star::beans::XPropertyChangeListener >&  ) throw(::com::sun::star::beans::UnknownPropertyException, ::com::sun::star::lang::WrappedTargetException, ::com::sun::star::uno::RuntimeException) {}
826 void SAL_CALL SdUnoSearchReplaceDescriptor::removePropertyChangeListener( const ::rtl::OUString& , const ::com::sun::star::uno::Reference< ::com::sun::star::beans::XPropertyChangeListener >&  ) throw(::com::sun::star::beans::UnknownPropertyException, ::com::sun::star::lang::WrappedTargetException, ::com::sun::star::uno::RuntimeException) {}
827 void SAL_CALL SdUnoSearchReplaceDescriptor::addVetoableChangeListener( const ::rtl::OUString& , const ::com::sun::star::uno::Reference< ::com::sun::star::beans::XVetoableChangeListener >&  ) throw(::com::sun::star::beans::UnknownPropertyException, ::com::sun::star::lang::WrappedTargetException, ::com::sun::star::uno::RuntimeException) {}
828 void SAL_CALL SdUnoSearchReplaceDescriptor::removeVetoableChangeListener( const ::rtl::OUString& , const ::com::sun::star::uno::Reference< ::com::sun::star::beans::XVetoableChangeListener >&  ) throw(::com::sun::star::beans::UnknownPropertyException, ::com::sun::star::lang::WrappedTargetException, ::com::sun::star::uno::RuntimeException) {}
829 
830 
831 /* ================================================================= */
832 
833 SdUnoFindAllAccess::SdUnoFindAllAccess( uno::Sequence< uno::Reference< uno::XInterface >  >& rSequence ) throw()
834 :maSequence( rSequence )
835 {
836 }
837 
838 SdUnoFindAllAccess::~SdUnoFindAllAccess() throw()
839 {
840 }
841 
842 // XElementAccess
843 uno::Type SAL_CALL SdUnoFindAllAccess::getElementType()
844 	throw(::com::sun::star::uno::RuntimeException)
845 {
846 	return ITYPE( text::XTextRange );
847 }
848 
849 sal_Bool SAL_CALL SdUnoFindAllAccess::hasElements()
850 	throw(::com::sun::star::uno::RuntimeException)
851 {
852 	return maSequence.getLength() > 0;
853 }
854 
855 // XIndexAccess
856 sal_Int32 SAL_CALL SdUnoFindAllAccess::getCount()
857 	throw(::com::sun::star::uno::RuntimeException)
858 {
859 	return maSequence.getLength();
860 }
861 
862 uno::Any SAL_CALL SdUnoFindAllAccess::getByIndex( sal_Int32 Index )
863 	throw(::com::sun::star::lang::IndexOutOfBoundsException, ::com::sun::star::lang::WrappedTargetException, ::com::sun::star::uno::RuntimeException)
864 {
865 	uno::Any aAny;
866 
867 	if( Index < 0 || Index >= getCount() )
868 		throw lang::IndexOutOfBoundsException();
869 
870 	const uno::Reference< uno::XInterface >  *pRefs = maSequence.getConstArray();
871 	if(pRefs)
872 		aAny <<= pRefs[ Index ];
873 	return aAny;
874 }
875 
876