xref: /trunk/main/sw/source/core/edit/edlingu.cxx (revision 7a52731c)
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_sw.hxx"
26 
27 #include <com/sun/star/linguistic2/ProofreadingResult.hpp>
28 #include <com/sun/star/linguistic2/XProofreader.hpp>
29 #include <com/sun/star/linguistic2/XProofreadingIterator.hpp>
30 #include <com/sun/star/text/XFlatParagraph.hpp>
31 
32 #include <unoflatpara.hxx>
33 
34 #include <comcore.hrc>
35 #include <hintids.hxx>
36 #include <linguistic/lngprops.hxx>
37 #include <vcl/msgbox.hxx>
38 #include <editeng/unolingu.hxx>
39 #include <editeng/svxacorr.hxx>
40 #include <editeng/langitem.hxx>
41 #include <editeng/SpellPortions.hxx>
42 #include <editeng/scripttypeitem.hxx>
43 #include <charatr.hxx>
44 #include <editsh.hxx>
45 #include <doc.hxx>
46 #include <IDocumentUndoRedo.hxx>
47 #include <rootfrm.hxx>      // SwRootFrm
48 #include <pam.hxx>
49 #include <swundo.hxx>		// fuer die UndoIds
50 #include <ndtxt.hxx>        // AdjHyphPos
51 #include <viewopt.hxx>      // HyphStart/End
52 #include <viscrs.hxx>		// SwShellCrsr
53 #include <SwGrammarMarkUp.hxx>		// SwWrongList
54 #include <mdiexp.hxx>		// Statusanzeige
55 #include <statstr.hrc>      // StatLine-String
56 #include <cntfrm.hxx>
57 #include <crsskip.hxx>
58 #include <splargs.hxx>
59 #include <redline.hxx>      // SwRedline
60 #include <docary.hxx>       // SwRedlineTbl
61 #include <docsh.hxx>
62 #include <txatbase.hxx>
63 #include <txtfrm.hxx>
64 
65 using namespace ::svx;
66 using namespace ::com::sun::star;
67 using namespace ::com::sun::star::uno;
68 using namespace ::com::sun::star::beans;
69 using namespace ::com::sun::star::linguistic2;
70 
71 #define C2U(cChar) rtl::OUString::createFromAscii(cChar)
72 
73 /*************************************************************************
74  *					   class SwLinguIter
75  *************************************************************************/
76 
77 class SwLinguIter
78 {
79 	SwEditShell *pSh;
80 	SwPosition	*pStart;
81 	SwPosition	*pEnd;
82 	SwPosition	*pCurr;
83 	SwPosition	*pCurrX;
84 	sal_uInt16 nCrsrCnt;
85 public:
86 	SwLinguIter();
87 
88 	inline SwEditShell *GetSh() 			{ return pSh; }
89 	inline const SwEditShell *GetSh() const { return pSh; }
90 
91 	inline const SwPosition *GetEnd() const { return pEnd; }
92 	inline void SetEnd( SwPosition* pNew ){ delete pEnd; pEnd = pNew; }
93 
94 	inline const SwPosition *GetStart() const { return pStart; }
95 	inline void SetStart( SwPosition* pNew ){ delete pStart; pStart = pNew; }
96 
97 	inline const SwPosition *GetCurr() const { return pCurr; }
98 	inline void SetCurr( SwPosition* pNew ){ delete pCurr; pCurr = pNew; }
99 
100 	inline const SwPosition *GetCurrX() const { return pCurrX; }
101 	inline void SetCurrX( SwPosition* pNew ){ delete pCurrX; pCurrX = pNew; }
102 
103 	inline sal_uInt16& GetCrsrCnt(){ return nCrsrCnt; }
104 
105 	// Der UI-Bauchladen:
106 	void _Start( SwEditShell *pSh, SwDocPositions eStart,
107                 SwDocPositions eEnd );
108     void _End(bool bRestoreSelection = true);
109 };
110 
111 /*************************************************************************
112  *					   class SwSpellIter
113  *************************************************************************/
114 
115 // #i18881# to be able to identify the postions of the changed words
116 // the content positions of each portion need to be saved
117 struct SpellContentPosition
118 {
119     sal_uInt16 nLeft;
120     sal_uInt16 nRight;
121 };
122 typedef std::vector<SpellContentPosition>  SpellContentPositions;
123 class SwSpellIter : public SwLinguIter
124 {
125 	uno::Reference< XSpellChecker1 > 	xSpeller;
126     ::svx::SpellPortions                aLastPortions;
127 
128     SpellContentPositions               aLastPositions;
129     bool                                bBackToStartOfSentence;
130     bool                                bMoveToEndOfSentence;
131 
132 
133     void    CreatePortion(uno::Reference< XSpellAlternatives > xAlt,
134                 linguistic2::ProofreadingResult* pGrammarResult,
135                 bool bIsField, bool bIsHidden);
136 
137     void    AddPortion(uno::Reference< XSpellAlternatives > xAlt,
138                        linguistic2::ProofreadingResult* pGrammarResult,
139                        const SpellContentPositions& rDeletedRedlines);
140 public:
141     SwSpellIter() :
142         bBackToStartOfSentence(false), bMoveToEndOfSentence(false) {}
143 
144 	void Start( SwEditShell *pSh, SwDocPositions eStart, SwDocPositions eEnd );
145 
146     uno::Any    Continue( sal_uInt16* pPageCnt, sal_uInt16* pPageSt );
147 
148     bool                                SpellSentence(::svx::SpellPortions& rPortions, bool bIsGrammarCheck);
149     void                                ToSentenceStart();
150     const ::svx::SpellPortions          GetLastPortions(){ return aLastPortions;}
151     SpellContentPositions               GetLastPositions() {return aLastPositions;}
152     void                                ContinueAfterThisSentence() { bMoveToEndOfSentence = true; }
153 };
154 
155 /*************************************************************************
156  *                     class SwConvIter
157  * used for text conversion
158  *************************************************************************/
159 
160 class SwConvIter : public SwLinguIter
161 {
162     SwConversionArgs &rArgs;
163 public:
164     SwConvIter( SwConversionArgs &rConvArgs ) :
165         rArgs( rConvArgs )
166     {}
167 
168     void Start( SwEditShell *pSh, SwDocPositions eStart, SwDocPositions eEnd );
169 
170     uno::Any    Continue( sal_uInt16* pPageCnt, sal_uInt16* pPageSt );
171 };
172 
173 /*************************************************************************
174  *					   class SwHyphIter
175  *************************************************************************/
176 
177 class SwHyphIter : public SwLinguIter
178 {
179 	sal_Bool bOldIdle;
180 	void DelSoftHyph( SwPaM &rPam );
181 
182 public:
183 	SwHyphIter() : bOldIdle(sal_False) {}
184 
185 	void Start( SwEditShell *pSh, SwDocPositions eStart, SwDocPositions eEnd );
186 	void End();
187 
188 	void Ignore();
189 
190     uno::Any    Continue( sal_uInt16* pPageCnt, sal_uInt16* pPageSt );
191 
192 	sal_Bool IsAuto();
193 	void InsertSoftHyph( const xub_StrLen nHyphPos );
194 	void ShowSelection();
195 };
196 
197 static SwSpellIter*	pSpellIter = 0;
198 static SwConvIter*  pConvIter = 0;
199 static SwHyphIter*	pHyphIter = 0;
200 
201 // Wir ersparen uns in Hyphenate ein GetFrm()
202 // Achtung: in txtedt.cxx stehen extern-Deklarationen auf diese Pointer!
203 const SwTxtNode *pLinguNode;
204 	  SwTxtFrm  *pLinguFrm;
205 
206 /*************************************************************************
207  *						SwLinguIter::SwLinguIter
208  *************************************************************************/
209 
210 SwLinguIter::SwLinguIter()
211 	: pSh( 0 ), pStart( 0 ), pEnd( 0 ), pCurr( 0 ), pCurrX( 0 )
212 {
213 	// @@@ es fehlt: Sicherstellen der Reentrance, ASSERTs etc.
214 }
215 
216 /*************************************************************************
217  *						SwLinguIter::Start
218  *************************************************************************/
219 
220 
221 
222 void SwLinguIter::_Start( SwEditShell *pShell, SwDocPositions eStart,
223                             SwDocPositions eEnd )
224 {
225 	// es fehlt: Sicherstellen der Reentrance, Locking
226 	if( pSh )
227 		return;
228 
229 	sal_Bool bSetCurr;
230 
231 	pSh = pShell;
232 
233 	SET_CURR_SHELL( pSh );
234 
235 	ASSERT( !pEnd, "LinguStart ohne End?");
236 
237 	SwPaM *pCrsr = pSh->GetCrsr();
238 
239 	// pStk->SetCurCrsr();
240 //	if( pCrsr->HasMark() || pCrsr != pCrsr->GetNext() )
241 	if( pShell->HasSelection() || pCrsr != pCrsr->GetNext() )
242 	{
243 		bSetCurr = 0 != GetCurr();
244 		nCrsrCnt = pSh->GetCrsrCnt();
245 		if( pSh->IsTableMode() )
246 			pSh->TblCrsrToCursor();
247 
248 		pSh->Push();
249 		sal_uInt16 n;
250 		for( n = 0; n < nCrsrCnt; ++n )
251 		{
252 			pSh->Push();
253 			pSh->DestroyCrsr();
254 		}
255 		pSh->Pop( sal_False );
256 	}
257 	else
258 	{
259 		bSetCurr = sal_False;
260 		nCrsrCnt = 1;
261 		pSh->Push();
262 		pSh->SetLinguRange( eStart, eEnd );
263 	}
264 
265 	pCrsr = pSh->GetCrsr();
266 	if ( *pCrsr->GetPoint() > *pCrsr->GetMark() )
267 		pCrsr->Exchange();
268 
269 	pStart = new SwPosition( *pCrsr->GetPoint() );
270 	pEnd = new SwPosition( *pCrsr->GetMark() );
271 	if( bSetCurr )
272 	{
273         SwPosition* pNew = new SwPosition( *GetStart() );
274 		SetCurr( pNew );
275 		pNew = new SwPosition( *pNew );
276 		SetCurrX( pNew );
277 	}
278 
279 	pCrsr->SetMark();
280 
281 	pLinguFrm = 0;
282 	pLinguNode = 0;
283 }
284 
285 /*************************************************************************
286  *						SwLinguIter::End
287  *************************************************************************/
288 
289 
290 
291 void SwLinguIter::_End(bool bRestoreSelection)
292 {
293 	if( !pSh )
294 		return;
295 
296 	ASSERT( pEnd, "SwEditShell::SpellEnd() ohne Start?");
297     if(bRestoreSelection)
298     {
299         while( nCrsrCnt-- )
300 		    pSh->Pop( sal_False );
301 
302 	    pSh->KillPams();
303 	    pSh->ClearMark();
304     }
305 	DELETEZ(pStart);
306 	DELETEZ(pEnd);
307 	DELETEZ(pCurr);
308 	DELETEZ(pCurrX);
309 
310 	pSh = 0;
311 
312 #ifdef LINGU_STATISTIK
313 	aSwLinguStat.Flush();
314 #endif
315 }
316 
317 /*************************************************************************
318  *				 virtual SwSpellIter::Start()
319  *************************************************************************/
320 
321 
322 
323 void SwSpellIter::Start( SwEditShell *pShell, SwDocPositions eStart,
324 						SwDocPositions eEnd )
325 {
326 	if( GetSh() )
327 		return;
328 
329  	uno::Reference< beans::XPropertySet >  xProp( ::GetLinguPropertySet() );
330 	xSpeller = ::GetSpellChecker();
331 	if ( xSpeller.is() )
332         _Start( pShell, eStart, eEnd );
333     aLastPortions.clear();
334     aLastPositions.clear();
335 }
336 
337 /*************************************************************************
338  *					 SwSpellIter::Continue
339  *************************************************************************/
340 
341 // SwSpellIter::Continue ist das alte Original von
342 // SwEditShell::SpellContinue()
343 
344 uno::Any SwSpellIter::Continue( sal_uInt16* pPageCnt, sal_uInt16* pPageSt )
345 {
346     //!!
347     //!! Please check SwConvIter also when modifying this
348     //!!
349 
350     uno::Any    aSpellRet;
351     SwEditShell *pMySh = GetSh();
352     if( !pMySh )
353         return aSpellRet;
354 
355 //	const SwPosition *pEnd = GetEnd();
356 
357 	ASSERT( GetEnd(), "SwEditShell::SpellContinue() ohne Start?");
358 
359     uno::Reference< uno::XInterface >  xSpellRet;
360 	sal_Bool bGoOn = sal_True;
361 	do {
362         SwPaM *pCrsr = pMySh->GetCrsr();
363 		if ( !pCrsr->HasMark() )
364 			pCrsr->SetMark();
365 
366 		uno::Reference< beans::XPropertySet >  xProp( GetLinguPropertySet() );
367         *pMySh->GetCrsr()->GetPoint() = *GetCurr();
368         *pMySh->GetCrsr()->GetMark() = *GetEnd();
369         pMySh->GetDoc()->Spell(*pMySh->GetCrsr(),
370                     xSpeller, pPageCnt, pPageSt, false ) >>= xSpellRet;
371 		bGoOn = GetCrsrCnt() > 1;
372 		if( xSpellRet.is() )
373 		{
374 			bGoOn = sal_False;
375 			SwPosition* pNewPoint = new SwPosition( *pCrsr->GetPoint() );
376 			SwPosition* pNewMark = new SwPosition( *pCrsr->GetMark() );
377             SetCurr( pNewPoint );
378             SetCurrX( pNewMark );
379 		}
380 		if( bGoOn )
381 		{
382             pMySh->Pop( sal_False );
383             pCrsr = pMySh->GetCrsr();
384 			if ( *pCrsr->GetPoint() > *pCrsr->GetMark() )
385 				pCrsr->Exchange();
386 			SwPosition* pNew = new SwPosition( *pCrsr->GetPoint() );
387 			SetStart( pNew );
388 			pNew = new SwPosition( *pCrsr->GetMark() );
389 			SetEnd( pNew );
390             pNew = new SwPosition( *GetStart() );
391 			SetCurr( pNew );
392 			pNew = new SwPosition( *pNew );
393 			SetCurrX( pNew );
394 			pCrsr->SetMark();
395 			--GetCrsrCnt();
396 		}
397 	}while ( bGoOn );
398     aSpellRet <<= xSpellRet;
399     return aSpellRet;
400 }
401 
402 /*************************************************************************
403  *               virtual SwConvIter::Start()
404  *************************************************************************/
405 
406 
407 
408 void SwConvIter::Start( SwEditShell *pShell, SwDocPositions eStart,
409                         SwDocPositions eEnd )
410 {
411     if( GetSh() )
412         return;
413     _Start( pShell, eStart, eEnd );
414 }
415 
416 /*************************************************************************
417  *                   SwConvIter::Continue
418  *************************************************************************/
419 
420 uno::Any SwConvIter::Continue( sal_uInt16* pPageCnt, sal_uInt16* pPageSt )
421 {
422     //!!
423     //!! Please check SwSpellIter also when modifying this
424     //!!
425 
426     uno::Any    aConvRet( makeAny( rtl::OUString() ) );
427     SwEditShell *pMySh = GetSh();
428     if( !pMySh )
429         return aConvRet;
430 
431 //  const SwPosition *pEnd = GetEnd();
432 
433     ASSERT( GetEnd(), "SwConvIter::Continue() ohne Start?");
434 
435     rtl::OUString aConvText;
436     sal_Bool bGoOn = sal_True;
437     do {
438         SwPaM *pCrsr = pMySh->GetCrsr();
439         if ( !pCrsr->HasMark() )
440             pCrsr->SetMark();
441 
442         *pMySh->GetCrsr()->GetPoint() = *GetCurr();
443         *pMySh->GetCrsr()->GetMark() = *GetEnd();
444 
445         // call function to find next text portion to be converted
446         uno::Reference< linguistic2::XSpellChecker1 > xEmpty;
447         pMySh->GetDoc()->Spell( *pMySh->GetCrsr(),
448                     xEmpty, pPageCnt, pPageSt, false, &rArgs ) >>= aConvText;
449 
450         bGoOn = GetCrsrCnt() > 1;
451         if( aConvText.getLength() )
452         {
453             bGoOn = sal_False;
454             SwPosition* pNewPoint = new SwPosition( *pCrsr->GetPoint() );
455             SwPosition* pNewMark = new SwPosition( *pCrsr->GetMark() );
456 
457             SetCurr( pNewPoint );
458             SetCurrX( pNewMark );
459         }
460         if( bGoOn )
461         {
462             pMySh->Pop( sal_False );
463             pCrsr = pMySh->GetCrsr();
464             if ( *pCrsr->GetPoint() > *pCrsr->GetMark() )
465                 pCrsr->Exchange();
466             SwPosition* pNew = new SwPosition( *pCrsr->GetPoint() );
467             SetStart( pNew );
468             pNew = new SwPosition( *pCrsr->GetMark() );
469             SetEnd( pNew );
470             pNew = new SwPosition( *GetStart() );
471             SetCurr( pNew );
472             pNew = new SwPosition( *pNew );
473             SetCurrX( pNew );
474             pCrsr->SetMark();
475             --GetCrsrCnt();
476         }
477     }while ( bGoOn );
478     return makeAny( aConvText );
479 }
480 
481 
482 /*************************************************************************
483  *                   SwHyphIter
484  *************************************************************************/
485 
486 
487 sal_Bool SwHyphIter::IsAuto()
488 {
489 	uno::Reference< beans::XPropertySet >  xProp( ::GetLinguPropertySet() );
490 	return xProp.is() ? *(sal_Bool*)xProp->getPropertyValue(
491                                 C2U(UPN_IS_HYPH_AUTO) ).getValue()
492 					  : sal_False;
493 }
494 
495 
496 void SwHyphIter::ShowSelection()
497 {
498     SwEditShell *pMySh = GetSh();
499     if( pMySh )
500 	{
501         pMySh->StartAction();
502 		// Ganz fatal: durch das EndAction() werden Formatierungen
503 		// angeregt, die dazu fuehren koennen, dass im Hyphenator
504 		// neue Worte eingestellt werden. Deswegen sichern!
505         pMySh->EndAction();
506 	}
507 }
508 
509 /*************************************************************************
510  *				 virtual SwHyphIter::Start()
511  *************************************************************************/
512 
513 
514 
515 void SwHyphIter::Start( SwEditShell *pShell, SwDocPositions eStart, SwDocPositions eEnd )
516 {
517 	// robust
518 	if( GetSh() || GetEnd() )
519 	{
520 		ASSERT( !GetSh(), "+SwEditShell::HyphStart: missing HyphEnd()" );
521 		return;
522 	}
523 
524 // nothing to be done (at least not in the way as in the "else" part)
525 	bOldIdle = pShell->GetViewOptions()->IsIdle();
526 	((SwViewOption*)pShell->GetViewOptions())->SetIdle( sal_False );
527 	_Start( pShell, eStart, eEnd );
528 }
529 
530 /*************************************************************************
531  *				   virtual SwHyphIter::End
532  *************************************************************************/
533 
534 // Selektionen wiederherstellen
535 
536 
537 
538 void SwHyphIter::End()
539 {
540 	if( !GetSh() )
541 		return;
542 	((SwViewOption*)GetSh()->GetViewOptions())->SetIdle( bOldIdle );
543 	_End();
544 }
545 
546 /*************************************************************************
547  *					 SwHyphIter::Continue
548  *************************************************************************/
549 
550 uno::Any SwHyphIter::Continue( sal_uInt16* pPageCnt, sal_uInt16* pPageSt )
551 {
552     uno::Any    aHyphRet;
553     SwEditShell *pMySh = GetSh();
554     if( !pMySh )
555         return aHyphRet;
556 
557 	const sal_Bool bAuto = IsAuto();
558 	 uno::Reference< XHyphenatedWord >  xHyphWord;
559 	sal_uInt16 nRet;
560 	sal_Bool bGoOn = sal_False;
561 	do {
562 		SwPaM *pCrsr;
563 		do {
564 			ASSERT( GetEnd(), "SwEditShell::SpellContinue() ohne Start?" );
565             pCrsr = pMySh->GetCrsr();
566 			if ( !pCrsr->HasMark() )
567 				pCrsr->SetMark();
568 			if ( *pCrsr->GetPoint() < *pCrsr->GetMark() )
569 			{
570 				pCrsr->Exchange();
571 				pCrsr->SetMark();
572 			}
573 
574 			// geraten BUG:
575 			if ( *pCrsr->End() > *GetEnd() )
576 				nRet = 0;
577 			else
578 			{
579 				*pCrsr->GetMark() = *GetEnd();
580 
581 				// Muss an der aktuellen Cursorpos das Wort getrennt werden ?
582                 const Point aCrsrPos( pMySh->GetCharRect().Pos() );
583                 xHyphWord = pMySh->GetDoc()->Hyphenate( pCrsr, aCrsrPos,
584 						 							  pPageCnt, pPageSt );
585 			}
586 
587 			if( bAuto && xHyphWord.is() )
588 			{
589                 pMySh->InsertSoftHyph( xHyphWord->getHyphenationPos() + 1);
590 			}
591 		} while( bAuto && xHyphWord.is() );	//end of do-while
592 		bGoOn = !xHyphWord.is() && GetCrsrCnt() > 1;
593 
594 		if( bGoOn )
595 		{
596             pMySh->Pop( sal_False );
597             pCrsr = pMySh->GetCrsr();
598 			if ( *pCrsr->GetPoint() > *pCrsr->GetMark() )
599 				pCrsr->Exchange();
600 			SwPosition* pNew = new SwPosition(*pCrsr->End());
601 			SetEnd( pNew );
602 			pCrsr->SetMark();
603 			--GetCrsrCnt();
604 		}
605 	} while ( bGoOn );
606     aHyphRet <<= xHyphWord;
607     return aHyphRet;
608 }
609 
610 /*************************************************************************
611  *					SwHyphIter::HyphIgnore
612  *************************************************************************/
613 
614 // Beschreibung: Trennstelle ignorieren
615 
616 void SwHyphIter::Ignore()
617 {
618     SwEditShell *pMySh = GetSh();
619     SwPaM *pCrsr = pMySh->GetCrsr();
620 
621 	// Alten SoftHyphen loeschen
622 	DelSoftHyph( *pCrsr );
623 
624 	// und weiter
625 	pCrsr->Start()->nContent = pCrsr->End()->nContent;
626 	pCrsr->SetMark();
627 }
628 
629 /*************************************************************************
630  *						  SwHyphIter::DelSoftHyph
631  *************************************************************************/
632 
633 void SwHyphIter::DelSoftHyph( SwPaM &rPam )
634 {
635 	const SwPosition* pStt = rPam.Start();
636 	const xub_StrLen nStart = pStt->nContent.GetIndex();
637 	const xub_StrLen nEnd   = rPam.End()->nContent.GetIndex();
638 	SwTxtNode *pNode = pStt->nNode.GetNode().GetTxtNode();
639 	pNode->DelSoftHyph( nStart, nEnd );
640 }
641 
642 /*************************************************************************
643  *					SwHyphIter::InsertSoftHyph
644  *************************************************************************/
645 
646 
647 void SwHyphIter::InsertSoftHyph( const xub_StrLen nHyphPos )
648 {
649     SwEditShell *pMySh = GetSh();
650     ASSERT( pMySh,  "+SwEditShell::InsertSoftHyph: missing HyphStart()");
651     if( !pMySh )
652 		return;
653 
654     SwPaM *pCrsr = pMySh->GetCrsr();
655     SwPosition* pSttPos = pCrsr->Start();
656     SwPosition* pEndPos = pCrsr->End();
657 
658 	xub_StrLen nLastHyphLen = GetEnd()->nContent.GetIndex() -
659                           pSttPos->nContent.GetIndex();
660 
661     if( pSttPos->nNode != pEndPos->nNode || !nLastHyphLen )
662 	{
663         ASSERT( pSttPos->nNode == pEndPos->nNode,
664 				"+SwEditShell::InsertSoftHyph: node warp during hyphenation" );
665 		ASSERT(nLastHyphLen, "+SwEditShell::InsertSoftHyph: missing HyphContinue()");
666         *pSttPos = *pEndPos;
667 		return;
668 	}
669 
670     pMySh->StartAction();
671 	{
672         SwDoc *pDoc = pMySh->GetDoc();
673 		DelSoftHyph( *pCrsr );
674         pSttPos->nContent += nHyphPos;
675         SwPaM aRg( *pSttPos );
676         pDoc->InsertString( aRg, CHAR_SOFTHYPHEN );
677 		// Durch das Einfuegen des SoftHyphs ist ein Zeichen hinzugekommen
678 //JP 18.07.95: warum, ist doch ein SwIndex, dieser wird doch mitverschoben !!
679 //        pSttPos->nContent++;
680 	}
681 	// Die Selektion wird wieder aufgehoben
682 	pCrsr->DeleteMark();
683     pMySh->EndAction();
684 	pCrsr->SetMark();
685 }
686 
687 // --------------------- Methoden der SwEditShell ------------------------
688 
689 bool SwEditShell::HasLastSentenceGotGrammarChecked() const
690 {
691     bool bTextWasGrammarChecked = false;
692     if (pSpellIter)
693     {
694         ::svx::SpellPortions aLastPortions( pSpellIter->GetLastPortions() );
695         for (size_t i = 0;  i < aLastPortions.size() && !bTextWasGrammarChecked;  ++i)
696         {
697             // bIsGrammarError is also true if the text was only checked but no
698             // grammar error was found. (That is if a ProofreadingResult was obtained in
699             // SwDoc::Spell and in turn bIsGrammarError was set in SwSpellIter::CreatePortion)
700             if (aLastPortions[i].bIsGrammarError)
701                 bTextWasGrammarChecked = true;
702         }
703     }
704     return bTextWasGrammarChecked;
705 }
706 
707 /*************************************************************************
708  *                      SwEditShell::HasConvIter
709  *************************************************************************/
710 
711 sal_Bool SwEditShell::HasConvIter() const
712 {
713     return 0 != pConvIter;
714 }
715 
716 /*************************************************************************
717  *                      SwEditShell::HasHyphIter
718  *************************************************************************/
719 
720 sal_Bool SwEditShell::HasHyphIter() const
721 {
722 	return 0 != pHyphIter;
723 }
724 
725 /*************************************************************************
726  *                      SwEditShell::SetFindRange
727  *************************************************************************/
728 
729 void SwEditShell::SetLinguRange( SwDocPositions eStart, SwDocPositions eEnd )
730 {
731 	SwPaM *pCrsr = GetCrsr();
732 	MakeFindRange( static_cast<sal_uInt16>(eStart), static_cast<sal_uInt16>(eEnd), pCrsr );
733 	if( *pCrsr->GetPoint() > *pCrsr->GetMark() )
734 		pCrsr->Exchange();
735 }
736 
737 /*************************************************************************
738  *                  SwEditShell::SpellStart
739  *************************************************************************/
740 
741 void SwEditShell::SpellStart(
742         SwDocPositions eStart, SwDocPositions eEnd, SwDocPositions eCurr,
743         SwConversionArgs *pConvArgs )
744 {
745     SwLinguIter *pLinguIter = 0;
746 
747 	// do not spell if interactive spelling is active elsewhere
748     if (!pConvArgs && !pSpellIter)
749 	{
750 		ASSERT( !pSpellIter, "wer ist da schon am spellen?" );
751 		pSpellIter = new SwSpellIter;
752         pLinguIter = pSpellIter;
753 	}
754     // do not do text conversion if it is active elsewhere
755     if (pConvArgs && !pConvIter)
756     {
757         ASSERT( !pConvIter, "text conversion already active!" );
758         pConvIter = new SwConvIter( *pConvArgs );
759         pLinguIter = pConvIter;
760     }
761 
762     if (pLinguIter)
763     {
764         SwCursor* pSwCrsr = GetSwCrsr();
765 
766         SwPosition *pTmp = new SwPosition( *pSwCrsr->GetPoint() );
767         pSwCrsr->FillFindPos( eCurr, *pTmp );
768         pLinguIter->SetCurr( pTmp );
769 
770         pTmp = new SwPosition( *pTmp );
771         pLinguIter->SetCurrX( pTmp );
772     }
773 
774     if (!pConvArgs && pSpellIter)
775         pSpellIter->Start( this, eStart, eEnd );
776     if (pConvArgs && pConvIter)
777         pConvIter->Start( this, eStart, eEnd );
778 }
779 
780 /*************************************************************************
781  *                  SwEditShell::SpellEnd
782  *************************************************************************/
783 
784 void SwEditShell::SpellEnd( SwConversionArgs *pConvArgs, bool bRestoreSelection )
785 {
786     if (!pConvArgs && pSpellIter && pSpellIter->GetSh() == this)
787 	{
788 		ASSERT( pSpellIter, "wo ist mein Iterator?" );
789 		pSpellIter->_End(bRestoreSelection);
790 		delete pSpellIter, pSpellIter = 0;
791 	}
792     if (pConvArgs && pConvIter && pConvIter->GetSh() == this)
793     {
794         ASSERT( pConvIter, "wo ist mein Iterator?" );
795         pConvIter->_End();
796         delete pConvIter, pConvIter = 0;
797     }
798 }
799 
800 /*************************************************************************
801  *                  SwEditShell::SpellContinue
802  *************************************************************************/
803 
804 // liefert Rueckgabewerte entsprechend SPL_ in splchk.hxx
805 
806 uno::Any SwEditShell::SpellContinue(
807         sal_uInt16* pPageCnt, sal_uInt16* pPageSt,
808         SwConversionArgs *pConvArgs )
809 {
810     uno::Any aRes;
811 
812     if ((!pConvArgs && pSpellIter->GetSh() != this) ||
813         ( pConvArgs && pConvIter->GetSh() != this))
814         return aRes;
815 
816 	if( pPageCnt && !*pPageCnt )
817 	{
818 		sal_uInt16 nEndPage = GetLayout()->GetPageNum();
819 		nEndPage += nEndPage * 10 / 100;
820 		*pPageCnt = nEndPage;
821 		if( nEndPage )
822 			::StartProgress( STR_STATSTR_SPELL, 0, nEndPage, GetDoc()->GetDocShell() );
823 	}
824 
825     ASSERT(  pConvArgs || pSpellIter, "SpellIter missing" );
826     ASSERT( !pConvArgs || pConvIter,  "ConvIter missing" );
827 	//JP 18.07.95: verhinder bei Fehlermeldungen die Anzeige der Selektionen
828 	//				KEIN StartAction, da damit auch die Paints abgeschaltet
829 	//				werden !!!!!
830 	++nStartAction;
831     rtl::OUString aRet;
832     uno::Reference< uno::XInterface >  xRet;
833     if (pConvArgs)
834     {
835         pConvIter->Continue( pPageCnt, pPageSt ) >>= aRet;
836         aRes <<= aRet;
837     }
838     else
839     {
840         pSpellIter->Continue( pPageCnt, pPageSt ) >>= xRet;
841         aRes <<= xRet;
842     }
843 	--nStartAction;
844 
845     if( aRet.getLength() || xRet.is() )
846 	{
847 		// dann die awt::Selection sichtbar machen
848 		StartAction();
849 		EndAction();
850 	}
851     return aRes;
852 }
853 /*************************************************************************
854  *					SwEditShell::HyphStart
855  *************************************************************************/
856 
857 /* Interaktive Trennung, BP 10.03.93
858  *
859  * 1) HyphStart
860  *    - Aufheben aller Selektionen
861  *    - Sichern des aktuellen Cursors
862  *	  - falls keine Selektion vorhanden:
863  *		- neue Selektion bis zum Dokumentende
864  * 2) HyphContinue
865  *	  - nLastHyphLen wird auf den Selektionsstart addiert
866  *	  - iteriert ueber alle selektierten Bereiche
867  *		- pDoc->Hyphenate() iteriert ueber alle Nodes der Selektion
868  *			- pTxtNode->Hyphenate() ruft das SwTxtFrm::Hyphenate zur EditShell
869  *				- SwTxtFrm:Hyphenate() iteriert ueber die Zeilen des Pams
870  *					- LineIter::Hyphenate() stellt den Hyphenator
871  *					  und den Pam auf das zu trennende Wort ein.
872  *	  - Es gibt nur zwei Returnwerte sal_True, wenn eine Trennstelle anliegt
873  *		und sal_False, wenn der Pam abgearbeitet wurde.
874  *	  - Bei sal_True wird das selektierte Wort zur Anzeige gebracht und
875  *		nLastHyphLen gesetzt.
876  *	  - Bei sal_False wird die aktuelle Selektion geloescht und die naechste
877  *		zur aktuellen gewaehlt. Return HYPH_OK, wenn keine mehr vorhanden.
878  * 3) InsertSoftHyph (wird ggf. von der UI gerufen)
879  *	  - Der aktuelle Cursor wird plaziert und das Attribut eingefuegt.
880  * 4) HyphEnd
881  *	  - Wiederherstellen des alten Cursors, EndAction
882  */
883 
884 
885 
886 void SwEditShell::HyphStart( SwDocPositions eStart, SwDocPositions eEnd )
887 {
888 	// do not hyphenate if interactive hyphenationg is active elsewhere
889 	if (!pHyphIter)
890 	{
891 		ASSERT( !pHyphIter, "wer ist da schon am hyphinieren?" );
892 		pHyphIter = new SwHyphIter;
893 		pHyphIter->Start( this, eStart, eEnd );
894 	}
895 }
896 
897 /*************************************************************************
898  *					SwEditShell::HyphEnd
899  *************************************************************************/
900 
901 // Selektionen wiederherstellen
902 
903 
904 
905 void SwEditShell::HyphEnd()
906 {
907 	if (pHyphIter->GetSh() == this)
908 	{
909 		ASSERT( pHyphIter, "wo ist mein Iterator?" );
910 		pHyphIter->End();
911 		delete pHyphIter, pHyphIter = 0;
912 	}
913 }
914 
915 /*************************************************************************
916  *					SwEditShell::HyphContinue
917  *************************************************************************/
918 
919 // Returnwerte: (BP: ich wuerde es genau umdrehen, aber die UI wuenscht es so)
920 // HYPH_CONTINUE, wenn eine Trennstelle anliegt
921 // HYPH_OK, wenn der selektierte Bereich abgearbeitet wurde.
922 
923 
924 uno::Reference< uno::XInterface >
925 	SwEditShell::HyphContinue( sal_uInt16* pPageCnt, sal_uInt16* pPageSt )
926 {
927 	if (pHyphIter->GetSh() != this)
928 		return 0;
929 
930 	if( pPageCnt && !*pPageCnt && !*pPageSt )
931 	{
932 		sal_uInt16 nEndPage = GetLayout()->GetPageNum();
933 		nEndPage += nEndPage * 10 / 100;
934 		if( nEndPage > 14 )
935 		{
936 			*pPageCnt = nEndPage;
937 			::StartProgress( STR_STATSTR_HYPHEN, 0, nEndPage, GetDoc()->GetDocShell());
938 		}
939 		else				// Hiermit unterdruecken wir ein fuer allemal
940 			*pPageSt = 1;	// das StatLineStartPercent
941 	}
942 
943 	ASSERT( pHyphIter, "wo ist mein Iterator?" );
944 	//JP 18.07.95: verhinder bei Fehlermeldungen die Anzeige der Selektionen
945 	//				KEIN StartAction, da damit auch die Paints abgeschaltet
946 	//				werden !!!!!
947 	++nStartAction;
948     uno::Reference< uno::XInterface >  xRet;
949     pHyphIter->Continue( pPageCnt, pPageSt ) >>= xRet;
950 	--nStartAction;
951 
952 	if( xRet.is() )
953 		pHyphIter->ShowSelection();
954 
955 	return xRet;
956 }
957 
958 
959 /*************************************************************************
960  *					SwEditShell::InsertSoftHyph
961  *************************************************************************/
962 
963 // Zum Einfuegen des SoftHyphens, Position ist der Offset
964 // innerhalb des getrennten Wortes.
965 
966 
967 void SwEditShell::InsertSoftHyph( const xub_StrLen nHyphPos )
968 {
969 	ASSERT( pHyphIter, "wo ist mein Iterator?" );
970 	pHyphIter->InsertSoftHyph( nHyphPos );
971 }
972 
973 
974 /*************************************************************************
975  *					SwEditShell::HyphIgnore
976  *************************************************************************/
977 
978 // Beschreibung: Trennstelle ignorieren
979 
980 void SwEditShell::HyphIgnore()
981 {
982 	ASSERT( pHyphIter, "wo ist mein Iterator?" );
983 	//JP 18.07.95: verhinder bei Fehlermeldungen die Anzeige der Selektionen
984 	//				KEIN StartAction, da damit auch die Paints abgeschaltet
985 	//				werden !!!!!
986 	++nStartAction;
987 	pHyphIter->Ignore();
988 	--nStartAction;
989 
990 	pHyphIter->ShowSelection();
991 }
992 
993 /*************************************************************************
994  *					SwEditShell::GetCorrection()
995  * liefert eine Liste von Vorschlaegen fuer falsch geschriebene Worte,
996  * ein NULL-Pointer signalisiert, dass das Wort richtig geschrieben ist,
997  * eine leere Liste, dass das Wort zwar unbekannt ist, aber keine Alternativen
998  * geliefert werden koennen.
999  *************************************************************************/
1000 
1001 
1002 uno::Reference< XSpellAlternatives >
1003     SwEditShell::GetCorrection( const Point* pPt, SwRect& rSelectRect )
1004 {
1005  	uno::Reference< XSpellAlternatives >  xSpellAlt;
1006 
1007 	if( IsTableMode() )
1008 		return NULL;
1009 	SwPaM* pCrsr = GetCrsr();
1010 	SwPosition aPos( *pCrsr->GetPoint() );
1011  	Point aPt( *pPt );
1012 	SwCrsrMoveState eTmpState( MV_SETONLYTEXT );
1013 	SwTxtNode *pNode;
1014 	SwWrongList *pWrong;
1015 	if( GetLayout()->GetCrsrOfst( &aPos, aPt, &eTmpState ) &&
1016 		0 != (pNode = aPos.nNode.GetNode().GetTxtNode()) &&
1017 		0 != (pWrong = pNode->GetWrong()) &&
1018 		!pNode->IsInProtectSect() )
1019 	{
1020 		xub_StrLen nBegin = aPos.nContent.GetIndex();
1021 		xub_StrLen nLen = 1;
1022 		if(	pWrong->InWrongWord(nBegin,nLen) && !pNode->IsSymbol(nBegin) )
1023 		{
1024 			String aText( pNode->GetTxt().Copy( nBegin, nLen ) );
1025 			String aWord( aText );
1026 			aWord.EraseAllChars( CH_TXTATR_BREAKWORD ).EraseAllChars( CH_TXTATR_INWORD );
1027 
1028             uno::Reference< XSpellChecker1 >  xSpell( ::GetSpellChecker() );
1029 			if( xSpell.is() )
1030 			{
1031                 LanguageType eActLang = (LanguageType)pNode->GetLang( nBegin, nLen );
1032 				if( xSpell->hasLanguage( eActLang ))
1033                 {
1034                     // restrict the maximal number of suggestions displayed
1035                     // in the context menu.
1036                     // Note: That could of course be done by clipping the
1037                     // resulting sequence but the current third party
1038                     // implementations result differs greatly if the number of
1039                     // suggestions to be retuned gets changed. Statistically
1040                     // it gets much better if told to return e.g. only 7 strings
1041                     // than returning e.g. 16 suggestions and using only the
1042                     // first 7. Thus we hand down the value to use to that
1043                     // implementation here by providing an additional parameter.
1044                     Sequence< PropertyValue > aPropVals(1);
1045                     PropertyValue &rVal = aPropVals.getArray()[0];
1046                     rVal.Name = C2U( UPN_MAX_NUMBER_OF_SUGGESTIONS );
1047                     rVal.Value <<= (sal_Int16) 7;
1048 
1049                     xSpellAlt = xSpell->spell( aWord, eActLang, aPropVals );
1050                 }
1051 			}
1052 
1053             if ( xSpellAlt.is() )   // error found?
1054 			{
1055                 //save the start and end positons of the line and the starting point
1056                 Push();
1057                 LeftMargin();
1058                 xub_StrLen nLineStart = GetCrsr()->GetPoint()->nContent.GetIndex();
1059                 RightMargin();
1060                 xub_StrLen nLineEnd = GetCrsr()->GetPoint()->nContent.GetIndex();
1061                 Pop(sal_False);
1062 
1063 				// make sure the selection build later from the
1064 				// data below does not include footnotes and other
1065 				// "in word" character to the left and right in order
1066 				// to preserve those. Therefore count those "in words"
1067 				// in order to modify the selection accordingly.
1068 				const sal_Unicode* pChar = aText.GetBuffer();
1069 				xub_StrLen nLeft = 0;
1070 				while (pChar && *pChar++ == CH_TXTATR_INWORD)
1071 					++nLeft;
1072 				pChar = aText.Len() ? aText.GetBuffer() + aText.Len() - 1 : 0;
1073 				xub_StrLen nRight = 0;
1074 				while (pChar && *pChar-- == CH_TXTATR_INWORD)
1075 					++nRight;
1076 
1077 				aPos.nContent = nBegin + nLeft;
1078                 pCrsr = GetCrsr();
1079 				*pCrsr->GetPoint() = aPos;
1080 				pCrsr->SetMark();
1081 				ExtendSelection( sal_True, nLen - nLeft - nRight );
1082                 //no determine the rectangle in the current line
1083                 xub_StrLen nWordStart = (nBegin + nLeft) < nLineStart ? nLineStart : nBegin + nLeft;
1084                 //take one less than the line end - otherwise the next line would be calculated
1085                 xub_StrLen nWordEnd = (nBegin + nLen - nLeft - nRight) > nLineEnd ? nLineEnd - 1: (nBegin + nLen - nLeft - nRight);
1086                 Push();
1087                 pCrsr->DeleteMark();
1088                 SwIndex& rContent = GetCrsr()->GetPoint()->nContent;
1089                 rContent = nWordStart;
1090                 SwRect aStartRect;
1091                 SwCrsrMoveState aState;
1092                 aState.bRealWidth = sal_True;
1093                 SwCntntNode* pCntntNode = pCrsr->GetCntntNode();
1094                 SwCntntFrm *pCntntFrame = pCntntNode->getLayoutFrm( GetLayout(), pPt, pCrsr->GetPoint(), sal_False);
1095 
1096                 pCntntFrame->GetCharRect( aStartRect, *pCrsr->GetPoint(), &aState );
1097                 rContent = nWordEnd;
1098                 SwRect aEndRect;
1099                 pCntntFrame->GetCharRect( aEndRect, *pCrsr->GetPoint(),&aState );
1100                 rSelectRect = aStartRect.Union( aEndRect );
1101                 Pop(sal_False);
1102             }
1103 		}
1104 	}
1105 	return xSpellAlt;
1106 }
1107 
1108 /*-------------------------------------------------------------------------
1109 
1110   -----------------------------------------------------------------------*/
1111 
1112 bool SwEditShell::GetGrammarCorrection(
1113     linguistic2::ProofreadingResult /*out*/ &rResult,    // the complete result
1114     sal_Int32 /*out*/ &rErrorPosInText,                     // offset of error position in string that was grammar checked...
1115     sal_Int32 /*out*/ &rErrorIndexInResult,                 // index of error in rResult.aGrammarErrors
1116     uno::Sequence< rtl::OUString > /*out*/ &rSuggestions,   // suggestions to be used for the error found
1117     const Point *pPt, SwRect &rSelectRect )
1118 {
1119     bool bRes = false;
1120 
1121     if( IsTableMode() )
1122         return bRes;
1123 
1124     SwPaM* pCrsr = GetCrsr();
1125     SwPosition aPos( *pCrsr->GetPoint() );
1126     Point aPt( *pPt );
1127     SwCrsrMoveState eTmpState( MV_SETONLYTEXT );
1128     SwTxtNode *pNode;
1129     SwGrammarMarkUp *pWrong;
1130     if( GetLayout()->GetCrsrOfst( &aPos, aPt, &eTmpState ) &&
1131         0 != (pNode = aPos.nNode.GetNode().GetTxtNode()) &&
1132         0 != (pWrong = pNode->GetGrammarCheck()) &&
1133         !pNode->IsInProtectSect() )
1134     {
1135         xub_StrLen nBegin = aPos.nContent.GetIndex();
1136         xub_StrLen nLen = 1;
1137         if (pWrong->InWrongWord(nBegin, nLen))
1138         {
1139             String aText( pNode->GetTxt().Copy( nBegin, nLen ) );
1140             String aWord( aText );
1141             aWord.EraseAllChars( CH_TXTATR_BREAKWORD ).EraseAllChars( CH_TXTATR_INWORD );
1142 
1143             uno::Reference< linguistic2::XProofreadingIterator >  xGCIterator( pDoc->GetGCIterator() );
1144             if (xGCIterator.is())
1145             {
1146 //                LanguageType eActLang = (LanguageType)pNode->GetLang( nBegin, nLen );
1147                 uno::Reference< lang::XComponent > xDoc( pDoc->GetDocShell()->GetBaseModel(), uno::UNO_QUERY );
1148 
1149                 // Expand the string:
1150                 rtl::OUString aExpandText;
1151                 const ModelToViewHelper::ConversionMap* pConversionMap =
1152                         pNode->BuildConversionMap( aExpandText );
1153                 // get XFlatParagraph to use...
1154                 uno::Reference< text::XFlatParagraph > xFlatPara = new SwXFlatParagraph( *pNode, aExpandText, pConversionMap );
1155 
1156                 // get error position of cursor in XFlatParagraph
1157                 rErrorPosInText = ModelToViewHelper::ConvertToViewPosition( pConversionMap, nBegin );
1158 
1159                 sal_Int32 nStartOfSentence = ModelToViewHelper::ConvertToViewPosition( pConversionMap, pWrong->getSentenceStart( nBegin ) );
1160                 sal_Int32 nEndOfSentence = ModelToViewHelper::ConvertToViewPosition( pConversionMap, pWrong->getSentenceEnd( nBegin ) );
1161                 if( nEndOfSentence == STRING_LEN )
1162                 {
1163 /*                    if( nStartOfSentence == 0 )
1164                     {
1165                         nStartOfSentence = -1;
1166                         nEndOfSentence = -1;
1167                     }
1168                     else */
1169                         nEndOfSentence = aExpandText.getLength();
1170                 }
1171 
1172                 rResult = xGCIterator->checkSentenceAtPosition(
1173                         xDoc, xFlatPara, aExpandText, lang::Locale(), nStartOfSentence, nEndOfSentence, rErrorPosInText );
1174                 bRes = true;
1175 
1176                 // get suggestions to use for the specific error position
1177                 sal_Int32 nErrors = rResult.aErrors.getLength();
1178                 rSuggestions.realloc( 0 );
1179                 for (sal_Int32 i = 0;  i < nErrors; ++i )
1180                 {
1181                     // return suggestions for first error that includes the given error position
1182                     const linguistic2::SingleProofreadingError &rError = rResult.aErrors[i];
1183                     if( (rError.nErrorStart <= rErrorPosInText) &&
1184                         (rErrorPosInText + nLen < rError.nErrorStart + rError.nErrorLength))
1185                     {
1186                         rSuggestions = rError.aSuggestions;
1187                         rErrorIndexInResult = i;
1188                         break;
1189                     }
1190                 }
1191             }
1192 
1193             if (rResult.aErrors.getLength() > 0)    // error found?
1194             {
1195                 //save the start and end positons of the line and the starting point
1196                 Push();
1197                 LeftMargin();
1198                 xub_StrLen nLineStart = GetCrsr()->GetPoint()->nContent.GetIndex();
1199                 RightMargin();
1200                 xub_StrLen nLineEnd = GetCrsr()->GetPoint()->nContent.GetIndex();
1201                 Pop(sal_False);
1202 
1203 #if OSL_DEBUG_LEVEL > 1
1204 //                pNode->GetGrammarCheck()->Invalidate( 0, STRING_LEN );
1205 //                pNode->SetGrammarCheckDirty( true );
1206 #endif
1207                 // make sure the selection build later from the
1208                 // data below does not include footnotes and other
1209                 // "in word" character to the left and right in order
1210                 // to preserve those. Therefore count those "in words"
1211                 // in order to modify the selection accordingly.
1212                 const sal_Unicode* pChar = aText.GetBuffer();
1213                 xub_StrLen nLeft = 0;
1214                 while (pChar && *pChar++ == CH_TXTATR_INWORD)
1215                     ++nLeft;
1216                 pChar = aText.Len() ? aText.GetBuffer() + aText.Len() - 1 : 0;
1217                 xub_StrLen nRight = 0;
1218                 while (pChar && *pChar-- == CH_TXTATR_INWORD)
1219                     ++nRight;
1220 
1221                 aPos.nContent = nBegin + nLeft;
1222                 pCrsr = GetCrsr();
1223                 *pCrsr->GetPoint() = aPos;
1224                 pCrsr->SetMark();
1225                 ExtendSelection( sal_True, nLen - nLeft - nRight );
1226                 //no determine the rectangle in the current line
1227                 xub_StrLen nWordStart = (nBegin + nLeft) < nLineStart ? nLineStart : nBegin + nLeft;
1228                 //take one less than the line end - otherwise the next line would be calculated
1229                 xub_StrLen nWordEnd = (nBegin + nLen - nLeft - nRight) > nLineEnd ? nLineEnd - 1: (nBegin + nLen - nLeft - nRight);
1230                 Push();
1231                 pCrsr->DeleteMark();
1232                 SwIndex& rContent = GetCrsr()->GetPoint()->nContent;
1233                 rContent = nWordStart;
1234                 SwRect aStartRect;
1235                 SwCrsrMoveState aState;
1236                 aState.bRealWidth = sal_True;
1237                 SwCntntNode* pCntntNode = pCrsr->GetCntntNode();
1238                 SwCntntFrm *pCntntFrame = pCntntNode->getLayoutFrm( GetLayout(), pPt, pCrsr->GetPoint(), sal_False);
1239 
1240                 pCntntFrame->GetCharRect( aStartRect, *pCrsr->GetPoint(), &aState );
1241                 rContent = nWordEnd;
1242                 SwRect aEndRect;
1243                 pCntntFrame->GetCharRect( aEndRect, *pCrsr->GetPoint(),&aState );
1244                 rSelectRect = aStartRect.Union( aEndRect );
1245                 Pop(sal_False);
1246             }
1247         }
1248     }
1249 
1250     return bRes;
1251 }
1252 
1253 /*-- 18.09.2003 15:08:18---------------------------------------------------
1254 
1255   -----------------------------------------------------------------------*/
1256 bool SwEditShell::SpellSentence(::svx::SpellPortions& rPortions, bool bIsGrammarCheck)
1257 {
1258     ASSERT(  pSpellIter, "SpellIter missing" );
1259     if(!pSpellIter)
1260         return false;
1261     bool bRet = pSpellIter->SpellSentence(rPortions, bIsGrammarCheck);
1262 
1263     // make Selection visible - this should simply move the
1264     // cursor to the end of the sentence
1265     StartAction();
1266     EndAction();
1267     return bRet;
1268 }
1269 /*-- 08.09.2008 09:35:19---------------------------------------------------
1270     make SpellIter start with the current sentence when called next time
1271   -----------------------------------------------------------------------*/
1272 void SwEditShell::PutSpellingToSentenceStart()
1273 {
1274     ASSERT(  pSpellIter, "SpellIter missing" );
1275     if(!pSpellIter)
1276         return;
1277     pSpellIter->ToSentenceStart();
1278 }
1279 /*-- 02.02.2005 14:34:41---------------------------------------------------
1280 
1281   -----------------------------------------------------------------------*/
1282 sal_uInt32 lcl_CountRedlines(
1283                             const ::svx::SpellPortions& rLastPortions)
1284 {
1285     sal_uInt32 nRet = 0;
1286     SpellPortions::const_iterator aIter = rLastPortions.begin();
1287     for( ; aIter != rLastPortions.end(); ++aIter)
1288     {
1289         if( aIter->bIsHidden )
1290             ++nRet;
1291     }
1292     return nRet;
1293 }
1294 /*-- 18.09.2003 15:08:20---------------------------------------------------
1295 
1296   -----------------------------------------------------------------------*/
1297 
1298 void SwEditShell::MoveContinuationPosToEndOfCheckedSentence()
1299 {
1300     // give hint that continuation position for spell/grammar checking is
1301     // at the end of this sentence
1302     if (pSpellIter)
1303     {
1304         pSpellIter->SetCurr( new SwPosition( *pSpellIter->GetCurrX() ) );
1305         pSpellIter->ContinueAfterThisSentence();
1306     }
1307 }
1308 
1309 
1310 void SwEditShell::ApplyChangedSentence(const ::svx::SpellPortions& rNewPortions, bool bRecheck)
1311 {
1312     // Note: rNewPortions.size() == 0 is valid and happens when the whole
1313     // sentence got removed in the dialog
1314 
1315     ASSERT(  pSpellIter, "SpellIter missing" );
1316     if(pSpellIter &&
1317        pSpellIter->GetLastPortions().size() > 0)    // no portions -> no text to be changed
1318     {
1319         const SpellPortions& rLastPortions = pSpellIter->GetLastPortions();
1320         const SpellContentPositions  rLastPositions = pSpellIter->GetLastPositions();
1321         ASSERT(rLastPortions.size() > 0 &&
1322                 rLastPortions.size() == rLastPositions.size(),
1323                 "last vectors of spelling results are not set or not equal")
1324 
1325         // iterate over the new portions, beginning at the end to take advantage of the previously
1326         // saved content positions
1327 
1328         pDoc->GetIDocumentUndoRedo().StartUndo( UNDO_OVERWRITE, NULL );
1329         StartAction();
1330 
1331         SwPaM *pCrsr = GetCrsr();
1332         // save cursor position (which should be at the end of the current sentence)
1333         // for later restoration
1334         Push();
1335 
1336         sal_uInt32 nRedlinePortions = lcl_CountRedlines(rLastPortions);
1337         if((rLastPortions.size() - nRedlinePortions) == rNewPortions.size())
1338         {
1339             DBG_ASSERT( rNewPortions.size() > 0, "rNewPortions should not be empty here" );
1340             DBG_ASSERT( rLastPortions.size() > 0, "rLastPortions should not be empty here" );
1341             DBG_ASSERT( rLastPositions.size() > 0, "rLastPositions should not be empty here" );
1342 
1343             //the simple case: the same number of elements on both sides
1344             //each changed element has to be applied to the corresponding source element
1345             svx::SpellPortions::const_iterator aCurrentNewPortion = rNewPortions.end();
1346             SpellPortions::const_iterator aCurrentOldPortion = rLastPortions.end();
1347             SpellContentPositions::const_iterator aCurrentOldPosition = rLastPositions.end();
1348             do
1349             {
1350                 --aCurrentNewPortion;
1351                 --aCurrentOldPortion;
1352                 --aCurrentOldPosition;
1353                 //jump over redline portions
1354                 while(aCurrentOldPortion->bIsHidden)
1355                 {
1356                     if (aCurrentOldPortion  != rLastPortions.begin() &&
1357                         aCurrentOldPosition != rLastPositions.begin())
1358                     {
1359                         --aCurrentOldPortion;
1360                         --aCurrentOldPosition;
1361                     }
1362                     else
1363                     {
1364                         DBG_ASSERT( 0, "ApplyChangedSentence: iterator positions broken" );
1365                         break;
1366                     }
1367                 }
1368 				if ( !pCrsr->HasMark() )
1369 					pCrsr->SetMark();
1370                 pCrsr->GetPoint()->nContent = aCurrentOldPosition->nLeft;
1371                 pCrsr->GetMark()->nContent = aCurrentOldPosition->nRight;
1372                 sal_uInt16 nScriptType = GetI18NScriptTypeOfLanguage( aCurrentNewPortion->eLanguage );
1373                 sal_uInt16 nLangWhichId = RES_CHRATR_LANGUAGE;
1374                 switch(nScriptType)
1375                 {
1376                     case SCRIPTTYPE_ASIAN : nLangWhichId = RES_CHRATR_CJK_LANGUAGE; break;
1377                     case SCRIPTTYPE_COMPLEX : nLangWhichId = RES_CHRATR_CTL_LANGUAGE; break;
1378                 }
1379                 if(aCurrentNewPortion->sText != aCurrentOldPortion->sText)
1380                 {
1381                     //change text ...
1382                     pDoc->DeleteAndJoin(*pCrsr);
1383                     // ... and apply language if necessary
1384                     if(aCurrentNewPortion->eLanguage != aCurrentOldPortion->eLanguage)
1385                         SetAttr( SvxLanguageItem(aCurrentNewPortion->eLanguage, nLangWhichId), nLangWhichId );
1386                     pDoc->InsertString(*pCrsr, aCurrentNewPortion->sText);
1387                 }
1388                 else if(aCurrentNewPortion->eLanguage != aCurrentOldPortion->eLanguage)
1389                 {
1390                     //apply language
1391                     SetAttr( SvxLanguageItem(aCurrentNewPortion->eLanguage, nLangWhichId), nLangWhichId );
1392                 }
1393                 else if( aCurrentNewPortion->bIgnoreThisError )
1394                 {
1395                     //add the 'ignore' markup to the TextNode's grammar ignore markup list
1396                     IgnoreGrammarErrorAt( *pCrsr );
1397                     DBG_ERROR("TODO: add ignore mark to text node");
1398                 }
1399                 if(aCurrentNewPortion == rNewPortions.begin())
1400                     break;
1401             }
1402             while(aCurrentNewPortion != rNewPortions.begin());
1403         }
1404         else
1405         {
1406             DBG_ASSERT( rLastPositions.size() > 0, "rLastPositions should not be empty here" );
1407 
1408             //select the complete sentence
1409             SpellContentPositions::const_iterator aCurrentEndPosition = rLastPositions.end();
1410             --aCurrentEndPosition;
1411             SpellContentPositions::const_iterator aCurrentStartPosition = rLastPositions.begin();
1412             pCrsr->GetPoint()->nContent = aCurrentStartPosition->nLeft;
1413             pCrsr->GetMark()->nContent = aCurrentEndPosition->nRight;
1414 
1415             //delete the sentence completely
1416             pDoc->DeleteAndJoin(*pCrsr);
1417             svx::SpellPortions::const_iterator aCurrentNewPortion = rNewPortions.begin();
1418             while(aCurrentNewPortion != rNewPortions.end())
1419             {
1420                 //set the language attribute
1421                 sal_uInt16 nScriptType = GetScriptType();
1422                 sal_uInt16 nLangWhichId = RES_CHRATR_LANGUAGE;
1423                 switch(nScriptType)
1424                 {
1425                     case SCRIPTTYPE_ASIAN : nLangWhichId = RES_CHRATR_CJK_LANGUAGE; break;
1426                     case SCRIPTTYPE_COMPLEX : nLangWhichId = RES_CHRATR_CTL_LANGUAGE; break;
1427                 }
1428                 SfxItemSet aSet(GetAttrPool(), nLangWhichId, nLangWhichId, 0);
1429                 GetCurAttr( aSet );
1430                 const SvxLanguageItem& rLang = static_cast<const SvxLanguageItem& >(aSet.Get(nLangWhichId));
1431                 if(rLang.GetLanguage() != aCurrentNewPortion->eLanguage)
1432 					SetAttr( SvxLanguageItem(aCurrentNewPortion->eLanguage, nLangWhichId) );
1433                 //insert the new string
1434                 pDoc->InsertString(*pCrsr, aCurrentNewPortion->sText);
1435 
1436                 //set the cursor to the end of the inserted string
1437                 *pCrsr->Start() = *pCrsr->End();
1438                 ++aCurrentNewPortion;
1439             }
1440         }
1441 
1442         // restore cursor to the end of the sentence
1443         // (will work also if the sentence length has changed,
1444         // since cursors get updated automatically!)
1445         Pop( sal_False );
1446 
1447         // collapse cursor to the end of the modified sentence
1448         *pCrsr->Start() = *pCrsr->End();
1449         if (bRecheck)
1450         {
1451             //in grammar check the current sentence has to be checked again
1452             GoStartSentence();
1453         }
1454         // set continuation position for spell/grammar checking to the end of this sentence
1455         pSpellIter->SetCurr( new SwPosition( *pCrsr->Start() ) );
1456 
1457         pDoc->GetIDocumentUndoRedo().EndUndo( UNDO_OVERWRITE, NULL );
1458         EndAction();
1459     }
1460 }
1461 /*-- 02.02.2005 10:46:45---------------------------------------------------
1462     collect all deleted redlines of the current text node beginning at the
1463     start of the cursor position
1464   -----------------------------------------------------------------------*/
1465 SpellContentPositions lcl_CollectDeletedRedlines(SwEditShell* pSh)
1466 {
1467     SpellContentPositions aRedlines;
1468     SwDoc* pDoc = pSh->GetDoc();
1469     const bool bShowChg = IDocumentRedlineAccess::IsShowChanges( pDoc->GetRedlineMode() );
1470     if ( bShowChg )
1471     {
1472         SwPaM *pCrsr = pSh->GetCrsr();
1473         const SwPosition* pStartPos = pCrsr->Start();
1474         const SwTxtNode* pTxtNode = pCrsr->GetNode()->GetTxtNode();
1475 
1476         sal_uInt16 nAct = pDoc->GetRedlinePos( *pTxtNode, USHRT_MAX );
1477         const xub_StrLen nStartIndex = pStartPos->nContent.GetIndex();
1478         for ( ; nAct < pDoc->GetRedlineTbl().Count(); nAct++ )
1479         {
1480             const SwRedline* pRed = pDoc->GetRedlineTbl()[ nAct ];
1481 
1482             if ( pRed->Start()->nNode > pTxtNode->GetIndex() )
1483                 break;
1484 
1485             if( nsRedlineType_t::REDLINE_DELETE == pRed->GetType() )
1486             {
1487                 xub_StrLen nStart, nEnd;
1488                 pRed->CalcStartEnd( pTxtNode->GetIndex(), nStart, nEnd );
1489                 if(nStart >= nStartIndex || nEnd >= nStartIndex)
1490                 {
1491                     SpellContentPosition aAdd;
1492                     aAdd.nLeft = nStart;
1493                     aAdd.nRight = nEnd;
1494                     aRedlines.push_back(aAdd);
1495                 }
1496             }
1497         }
1498     }
1499     return aRedlines;
1500 }
1501 /*-- 02.02.2005 11:06:12---------------------------------------------------
1502     remove the redline positions after the current selection
1503   -----------------------------------------------------------------------*/
1504 void lcl_CutRedlines( SpellContentPositions& aDeletedRedlines, SwEditShell* pSh )
1505 {
1506     if(!aDeletedRedlines.empty())
1507     {
1508         SwPaM *pCrsr = pSh->GetCrsr();
1509         const SwPosition* pEndPos = pCrsr->End();
1510         xub_StrLen nEnd = pEndPos->nContent.GetIndex();
1511         while(!aDeletedRedlines.empty() &&
1512                 aDeletedRedlines.back().nLeft > nEnd)
1513         {
1514             aDeletedRedlines.pop_back();
1515         }
1516     }
1517 }
1518 /*-- 02.02.2005 11:43:00---------------------------------------------------
1519 
1520   -----------------------------------------------------------------------*/
1521 SpellContentPosition  lcl_FindNextDeletedRedline(
1522         const SpellContentPositions& rDeletedRedlines,
1523         xub_StrLen nSearchFrom )
1524 {
1525     SpellContentPosition aRet;
1526     aRet.nLeft = aRet.nRight = STRING_MAXLEN;
1527     if(!rDeletedRedlines.empty())
1528     {
1529         SpellContentPositions::const_iterator aIter = rDeletedRedlines.begin();
1530         for( ; aIter != rDeletedRedlines.end(); ++aIter)
1531         {
1532             if(aIter->nLeft < nSearchFrom)
1533                 continue;
1534             aRet = *aIter;
1535             break;
1536         }
1537     }
1538     return aRet;
1539 }
1540 /*-- 18.09.2003 15:08:20---------------------------------------------------
1541 
1542   -----------------------------------------------------------------------*/
1543 bool SwSpellIter::SpellSentence(::svx::SpellPortions& rPortions, bool bIsGrammarCheck)
1544 {
1545     bool bRet = false;
1546     aLastPortions.clear();
1547     aLastPositions.clear();
1548 
1549     SwEditShell *pMySh = GetSh();
1550     if( !pMySh )
1551         return false;
1552 
1553     ASSERT( GetEnd(), "SwEditShell::SpellSentence() ohne Start?");
1554 
1555     uno::Reference< XSpellAlternatives >  xSpellRet;
1556     linguistic2::ProofreadingResult aGrammarResult;
1557     sal_Bool bGoOn = sal_True;
1558     bool bGrammarErrorFound = false;
1559     do {
1560         SwPaM *pCrsr = pMySh->GetCrsr();
1561         if ( !pCrsr->HasMark() )
1562             pCrsr->SetMark();
1563 
1564         *pCrsr->GetPoint() = *GetCurr();
1565         *pCrsr->GetMark() = *GetEnd();
1566 
1567         if( bBackToStartOfSentence )
1568         {
1569             pMySh->GoStartSentence();
1570             bBackToStartOfSentence = false;
1571         }
1572         uno::Any aSpellRet =
1573         pMySh->GetDoc()->Spell(*pCrsr,
1574                     xSpeller, 0, 0, bIsGrammarCheck );
1575         aSpellRet >>= xSpellRet;
1576         aSpellRet >>= aGrammarResult;
1577         bGoOn = GetCrsrCnt() > 1;
1578         bGrammarErrorFound = aGrammarResult.aErrors.getLength() > 0;
1579         if( xSpellRet.is() || bGrammarErrorFound )
1580         {
1581             bGoOn = sal_False;
1582             SwPosition* pNewPoint = new SwPosition( *pCrsr->GetPoint() );
1583             SwPosition* pNewMark = new SwPosition( *pCrsr->GetMark() );
1584 
1585             SetCurr( pNewPoint );
1586             SetCurrX( pNewMark );
1587         }
1588         if( bGoOn )
1589         {
1590             pMySh->Pop( sal_False );
1591             pCrsr = pMySh->GetCrsr();
1592             if ( *pCrsr->GetPoint() > *pCrsr->GetMark() )
1593                 pCrsr->Exchange();
1594             SwPosition* pNew = new SwPosition( *pCrsr->GetPoint() );
1595             SetStart( pNew );
1596             pNew = new SwPosition( *pCrsr->GetMark() );
1597             SetEnd( pNew );
1598             pNew = new SwPosition( *GetStart() );
1599             SetCurr( pNew );
1600             pNew = new SwPosition( *pNew );
1601             SetCurrX( pNew );
1602             pCrsr->SetMark();
1603             --GetCrsrCnt();
1604         }
1605     }
1606     while ( bGoOn );
1607     if(xSpellRet.is() || bGrammarErrorFound)
1608     {
1609         //an error has been found
1610         //To fill the spell portions the beginning of the sentence has to be found
1611         SwPaM *pCrsr = pMySh->GetCrsr();
1612         //set the mark to the right if necessary
1613         if ( *pCrsr->GetPoint() > *pCrsr->GetMark() )
1614             pCrsr->Exchange();
1615         //the cursor has to be collapsed on the left to go to the start of the sentence - if sentence ends inside of the error
1616         pCrsr->DeleteMark();
1617         pCrsr->SetMark();
1618         sal_Bool bStartSent = 0 != pMySh->GoStartSentence();
1619         SpellContentPositions aDeletedRedlines = lcl_CollectDeletedRedlines(pMySh);
1620         if(bStartSent)
1621         {
1622             //create a portion from the start part
1623             AddPortion(0, 0, aDeletedRedlines);
1624         }
1625         //Set the cursor to the error already found
1626         *pCrsr->GetPoint() = *GetCurrX();
1627         *pCrsr->GetMark() = *GetCurr();
1628         AddPortion(xSpellRet, &aGrammarResult, aDeletedRedlines);
1629 
1630 
1631         //save the end position of the error to continue from here
1632         SwPosition aSaveStartPos = *pCrsr->End();
1633         //determine the end of the current sentence
1634         if ( *pCrsr->GetPoint() < *pCrsr->GetMark() )
1635             pCrsr->Exchange();
1636         //again collapse to start marking after the end of the error
1637         pCrsr->DeleteMark();
1638         pCrsr->SetMark();
1639 
1640         pMySh->GoEndSentence();
1641         if( bGrammarErrorFound )
1642         {
1643             rtl::OUString aExpandText;
1644             const ModelToViewHelper::ConversionMap* pConversionMap = ((SwTxtNode*)pCrsr->GetNode())->BuildConversionMap( aExpandText );
1645             xub_StrLen nSentenceEnd = (xub_StrLen)ModelToViewHelper::ConvertToViewPosition( pConversionMap, aGrammarResult.nBehindEndOfSentencePosition );
1646             // remove trailing space
1647             if( aExpandText[nSentenceEnd - 1] == ' ' )
1648                 --nSentenceEnd;
1649             if( pCrsr->End()->nContent.GetIndex() < nSentenceEnd )
1650             {
1651                 pCrsr->End()->nContent.Assign(
1652                     pCrsr->End()->nNode.GetNode().GetCntntNode(), nSentenceEnd);
1653             }
1654         }
1655 
1656         lcl_CutRedlines( aDeletedRedlines, pMySh );
1657         //save the 'global' end of the spellchecking
1658         const SwPosition aSaveEndPos = *GetEnd();
1659         //set the sentence end as 'local' end
1660         SetEnd( new SwPosition( *pCrsr->End() ));
1661 
1662         *pCrsr->GetPoint() = aSaveStartPos;
1663         *pCrsr->GetMark() = *GetEnd();
1664         //now the rest of the sentence has to be searched for errors
1665         // for each error the non-error text between the current and the last error has
1666         // to be added to the portions - if necessary broken into same-language-portions
1667         if( !bGrammarErrorFound ) //in grammar check there's only one error returned
1668         {
1669             do
1670             {
1671                 xSpellRet = 0;
1672                 // don't search for grammar errors here anymore!
1673                 pMySh->GetDoc()->Spell(*pCrsr,
1674                             xSpeller, 0, 0, false ) >>= xSpellRet;
1675                 if ( *pCrsr->GetPoint() > *pCrsr->GetMark() )
1676                     pCrsr->Exchange();
1677                 SetCurr( new SwPosition( *pCrsr->GetPoint() ));
1678                 SetCurrX( new SwPosition( *pCrsr->GetMark() ));
1679 
1680                 //if an error has been found go back to the text
1681                 //preceeding the error
1682                 if(xSpellRet.is())
1683                 {
1684                     *pCrsr->GetPoint() = aSaveStartPos;
1685                     *pCrsr->GetMark() = *GetCurr();
1686                 }
1687                 //add the portion
1688                 AddPortion(0, 0, aDeletedRedlines);
1689 
1690                 if(xSpellRet.is())
1691                 {
1692                     *pCrsr->GetPoint() = *GetCurr();
1693                     *pCrsr->GetMark() = *GetCurrX();
1694                     AddPortion(xSpellRet, 0, aDeletedRedlines);
1695                     //move the cursor to the end of the error string
1696                     *pCrsr->GetPoint() = *GetCurrX();
1697                     //and save the end of the error as new start position
1698                     aSaveStartPos = *GetCurrX();
1699                     //and the end of the sentence
1700                     *pCrsr->GetMark() = *GetEnd();
1701                 }
1702                 // if the end of the sentence has already been reached then break here
1703                 if(*GetCurrX() >= *GetEnd())
1704                     break;
1705             }
1706             while(xSpellRet.is());
1707         }
1708         else
1709         {
1710             //go to the end of sentence as the grammar check returned it
1711             // at this time the Point is behind the grammar error
1712             // and the mark points to the sentence end as
1713             if ( *pCrsr->GetPoint() < *pCrsr->GetMark() )
1714                 pCrsr->Exchange();
1715         }
1716 
1717         // the part between the last error and the end of the sentence has to be added
1718         *pMySh->GetCrsr()->GetPoint() = *GetEnd();
1719         if(*GetCurrX() < *GetEnd())
1720         {
1721             AddPortion(0, 0, aDeletedRedlines);
1722         }
1723         //set the shell cursor to the end of the sentence to prevent a visible selection
1724         *pCrsr->GetMark() = *GetEnd();
1725 		if( !bIsGrammarCheck )
1726         {
1727             //set the current position to the end of the sentence
1728             SetCurr( new SwPosition(*GetEnd()) );
1729         }
1730         //restore the 'global' end
1731         SetEnd( new SwPosition(aSaveEndPos) );
1732         rPortions = aLastPortions;
1733         bRet = true;
1734     }
1735     else
1736     {
1737         //if no error could be found the selection has to be corrected - at least if it's not in the body
1738         *pMySh->GetCrsr()->GetPoint() = *GetEnd();
1739         pMySh->GetCrsr()->DeleteMark();
1740     }
1741 
1742     return bRet;
1743 }
1744 
1745 /*-- 08.09.2008 09:37:15---------------------------------------------------
1746 
1747   -----------------------------------------------------------------------*/
1748 void SwSpellIter::ToSentenceStart()
1749 {
1750     bBackToStartOfSentence = true;
1751 }
1752 /*-- 08.10.2003 08:49:56---------------------------------------------------
1753 
1754   -----------------------------------------------------------------------*/
1755 LanguageType lcl_GetLanguage(SwEditShell& rSh)
1756 {
1757     sal_uInt16 nScriptType = rSh.GetScriptType();
1758     sal_uInt16 nLangWhichId = RES_CHRATR_LANGUAGE;
1759 
1760     switch(nScriptType)
1761     {
1762         case SCRIPTTYPE_ASIAN : nLangWhichId = RES_CHRATR_CJK_LANGUAGE; break;
1763         case SCRIPTTYPE_COMPLEX : nLangWhichId = RES_CHRATR_CTL_LANGUAGE; break;
1764     }
1765     SfxItemSet aSet(rSh.GetAttrPool(), nLangWhichId, nLangWhichId, 0);
1766     rSh.GetCurAttr( aSet );
1767     const SvxLanguageItem& rLang = static_cast<const SvxLanguageItem& >(aSet.Get(nLangWhichId));
1768     return rLang.GetLanguage();
1769 }
1770 /*-- 08.10.2003 08:53:27---------------------------------------------------
1771     create a text portion at the given position
1772   -----------------------------------------------------------------------*/
1773 void SwSpellIter::CreatePortion(uno::Reference< XSpellAlternatives > xAlt,
1774                         linguistic2::ProofreadingResult* pGrammarResult,
1775         bool bIsField, bool bIsHidden)
1776 {
1777     svx::SpellPortion aPortion;
1778     String sText;
1779     GetSh()->GetSelectedText( sText );
1780 	if(sText.Len())
1781 	{
1782         //in case of redlined deletions the selection of an error is not
1783         //the same as the _real_ word
1784         if(xAlt.is())
1785             aPortion.sText = xAlt->getWord();
1786         else if(pGrammarResult)
1787         {
1788             aPortion.bIsGrammarError = true;
1789             if(pGrammarResult->aErrors.getLength())
1790             {
1791                 aPortion.aGrammarError = pGrammarResult->aErrors[0];
1792                 aPortion.sText = pGrammarResult->aText.copy( aPortion.aGrammarError.nErrorStart, aPortion.aGrammarError.nErrorLength );
1793                 aPortion.xGrammarChecker = pGrammarResult->xProofreader;
1794                 const beans::PropertyValue* pProperties = pGrammarResult->aProperties.getConstArray();
1795                 for( sal_Int32 nProp = 0; nProp < pGrammarResult->aProperties.getLength(); ++nProp )
1796                 {
1797                     if( pProperties->Name.equalsAscii("DialogTitle") )
1798                     {
1799                         pProperties->Value >>= aPortion.sDialogTitle;
1800                         break;
1801                     }
1802                 }
1803             }
1804         }
1805         else
1806             aPortion.sText = sText;
1807 		aPortion.eLanguage = lcl_GetLanguage(*GetSh());
1808 		aPortion.bIsField = bIsField;
1809         aPortion.bIsHidden = bIsHidden;
1810 		aPortion.xAlternatives = xAlt;
1811 		SpellContentPosition aPosition;
1812 		SwPaM *pCrsr = GetSh()->GetCrsr();
1813 		aPosition.nLeft = pCrsr->Start()->nContent.GetIndex();
1814 		aPosition.nRight = pCrsr->End()->nContent.GetIndex();
1815 		aLastPortions.push_back(aPortion);
1816 		aLastPositions.push_back(aPosition);
1817 	}
1818 }
1819 /*-- 19.09.2003 13:05:43---------------------------------------------------
1820 
1821   -----------------------------------------------------------------------*/
1822 void    SwSpellIter::AddPortion(uno::Reference< XSpellAlternatives > xAlt,
1823                                 linguistic2::ProofreadingResult* pGrammarResult,
1824                                 const SpellContentPositions& rDeletedRedlines)
1825 {
1826     SwEditShell *pMySh = GetSh();
1827     String sText;
1828     pMySh->GetSelectedText( sText );
1829     if(sText.Len())
1830     {
1831         if(xAlt.is() || pGrammarResult != 0)
1832         {
1833             CreatePortion(xAlt, pGrammarResult, false, false);
1834         }
1835         else
1836         {
1837             SwPaM *pCrsr = GetSh()->GetCrsr();
1838             if ( *pCrsr->GetPoint() > *pCrsr->GetMark() )
1839                 pCrsr->Exchange();
1840             //save the start and end positions
1841             SwPosition aStart(*pCrsr->GetPoint());
1842             SwPosition aEnd(*pCrsr->GetMark());
1843             //iterate over the text to find changes in language
1844             //set the mark equal to the point
1845             *pCrsr->GetMark() = aStart;
1846             SwTxtNode* pTxtNode = pCrsr->GetNode()->GetTxtNode();
1847             LanguageType eStartLanguage = lcl_GetLanguage(*GetSh());
1848             SpellContentPosition  aNextRedline = lcl_FindNextDeletedRedline(
1849                         rDeletedRedlines, aStart.nContent.GetIndex() );
1850             if( aNextRedline.nLeft == aStart.nContent.GetIndex() )
1851             {
1852                 //select until the end of the current redline
1853                 xub_StrLen nEnd = aEnd.nContent.GetIndex() < aNextRedline.nRight ?
1854                             aEnd.nContent.GetIndex() : aNextRedline.nRight;
1855                 pCrsr->GetPoint()->nContent.Assign( pTxtNode, nEnd );
1856                 CreatePortion(xAlt, pGrammarResult, false, true);
1857                 aStart = *pCrsr->End();
1858                 //search for next redline
1859                 aNextRedline = lcl_FindNextDeletedRedline(
1860                             rDeletedRedlines, aStart.nContent.GetIndex() );
1861             }
1862             while(*pCrsr->GetPoint() < aEnd)
1863             {
1864                 //#125786 in table cell with fixed row height the cursor might not move forward
1865                 if(!GetSh()->Right(1, CRSR_SKIP_CELLS))
1866                     break;
1867 
1868                 bool bField = false;
1869                 //read the character at the current position to check if it's a field
1870                 xub_Unicode cChar = pTxtNode->GetTxt().GetChar( pCrsr->GetMark()->nContent.GetIndex() );
1871                 if( CH_TXTATR_BREAKWORD == cChar || CH_TXTATR_INWORD == cChar)
1872                 {
1873                     const SwTxtAttr* pTxtAttr = pTxtNode->GetTxtAttrForCharAt(
1874                         pCrsr->GetMark()->nContent.GetIndex() );
1875                     const sal_uInt16 nWhich = pTxtAttr
1876                         ? pTxtAttr->Which()
1877                         : static_cast<sal_uInt16>(RES_TXTATR_END);
1878                     switch (nWhich)
1879                     {
1880                         case RES_TXTATR_FIELD:
1881                         case RES_TXTATR_FTN:
1882                         case RES_TXTATR_FLYCNT:
1883                             bField = true;
1884                             break;
1885                     }
1886                 }
1887 
1888                 LanguageType eCurLanguage = lcl_GetLanguage(*GetSh());
1889                 bool bRedline = aNextRedline.nLeft == pCrsr->GetPoint()->nContent.GetIndex();
1890                 // create a portion if the next character
1891                 //  - is a field,
1892                 //  - is at the beginning of a deleted redline
1893                 //  - has a different language
1894                 if(bField || bRedline || eCurLanguage != eStartLanguage)
1895                 {
1896                     eStartLanguage = eCurLanguage;
1897                     //go one step back - the cursor currently selects the first character
1898                     //with a different language
1899                     //in the case of redlining it's different
1900                     if(eCurLanguage != eStartLanguage || bField)
1901                         *pCrsr->GetPoint() = *pCrsr->GetMark();
1902                     //set to the last start
1903                     *pCrsr->GetMark() = aStart;
1904                     //create portion should only be called if a selection exists
1905                     //there's no selection if there's a field at the beginning
1906                     if(*pCrsr->Start() != *pCrsr->End())
1907                         CreatePortion(xAlt, pGrammarResult, false, false);
1908                     aStart = *pCrsr->End();
1909                     //now export the field - if there is any
1910                     if(bField)
1911                     {
1912                         *pCrsr->GetMark() = *pCrsr->GetPoint();
1913                         GetSh()->Right(1, CRSR_SKIP_CELLS);
1914                         CreatePortion(xAlt, pGrammarResult, true, false);
1915                         aStart = *pCrsr->End();
1916                     }
1917                 }
1918                 // if a redline start then create a portion for it
1919                 if(bRedline)
1920                 {
1921                     *pCrsr->GetMark() = *pCrsr->GetPoint();
1922                     //select until the end of the current redline
1923                     xub_StrLen nEnd = aEnd.nContent.GetIndex() < aNextRedline.nRight ?
1924                                 aEnd.nContent.GetIndex() : aNextRedline.nRight;
1925                     pCrsr->GetPoint()->nContent.Assign( pTxtNode, nEnd );
1926                     CreatePortion(xAlt, pGrammarResult, false, true);
1927                     aStart = *pCrsr->End();
1928                     //search for next redline
1929                     aNextRedline = lcl_FindNextDeletedRedline(
1930                                 rDeletedRedlines, aStart.nContent.GetIndex() );
1931                 }
1932                 *pCrsr->GetMark() = *pCrsr->GetPoint();
1933             }
1934             pCrsr->SetMark();
1935             *pCrsr->GetMark() = aStart;
1936             CreatePortion(xAlt, pGrammarResult, false, false);
1937         }
1938     }
1939 }
1940 /*-- 07.08.2008 15:01:25---------------------------------------------------
1941 
1942   -----------------------------------------------------------------------*/
1943 void SwEditShell::IgnoreGrammarErrorAt( SwPaM& rErrorPosition )
1944 {
1945     SwTxtNode *pNode;
1946     SwWrongList *pWrong;
1947     SwNodeIndex aIdx = rErrorPosition.Start()->nNode;
1948     SwNodeIndex aEndIdx = rErrorPosition.Start()->nNode;
1949     xub_StrLen nStart = rErrorPosition.Start()->nContent.GetIndex();
1950     xub_StrLen nEnd = STRING_LEN;
1951     while( aIdx <= aEndIdx )
1952     {
1953         pNode = aIdx.GetNode().GetTxtNode();
1954         if( pNode ) {
1955             if( aIdx == aEndIdx )
1956                 nEnd = rErrorPosition.End()->nContent.GetIndex();
1957             pWrong = pNode->GetGrammarCheck();
1958             if( pWrong )
1959                 pWrong->RemoveEntry( nStart, nEnd );
1960             pWrong = pNode->GetWrong();
1961             if( pWrong )
1962                 pWrong->RemoveEntry( nStart, nEnd );
1963             SwTxtFrm::repaintTextFrames( *pNode );
1964         }
1965         ++aIdx;
1966         nStart = 0;
1967     }
1968 }
1969 
1970 
1971