xref: /aoo42x/main/svtools/source/edit/textview.cxx (revision 5900e8ec)
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_svtools.hxx"
26 #include <svtools/textview.hxx>
27 #include <svtools/texteng.hxx>
28 #include <textdoc.hxx>
29 #include <svtools/textdata.hxx>
30 #include <textdat2.hxx>
31 
32 #include <svl/undo.hxx>
33 #include <vcl/cursor.hxx>
34 #include <vcl/window.hxx>
35 #include <vcl/svapp.hxx>
36 #include <vcl/sound.hxx>
37 #include <tools/stream.hxx>
38 
39 #include <sot/formats.hxx>
40 #include <svl/urlbmk.hxx>
41 
42 #ifndef _COM_SUN_STAR_TEXT_XBREAKITERATOR_HPP_
43 #include <com/sun/star/i18n/XBreakIterator.hpp>
44 #endif
45 
46 #ifndef _COM_SUN_STAR_TEXT_CHARACTERITERATORMODE_HPP_
47 #include <com/sun/star/i18n/CharacterIteratorMode.hpp>
48 #endif
49 
50 #ifndef _COM_SUN_STAR_TEXT_WORDTYPE_HPP_
51 #include <com/sun/star/i18n/WordType.hpp>
52 #endif
53 #include <cppuhelper/weak.hxx>
54 #include <vcl/unohelp.hxx>
55 #include <com/sun/star/datatransfer/XTransferable.hpp>
56 #include <com/sun/star/datatransfer/clipboard/XClipboard.hpp>
57 #include <com/sun/star/datatransfer/clipboard/XFlushableClipboard.hpp>
58 #include <com/sun/star/lang/XMultiServiceFactory.hpp>
59 
60 #ifndef _COM_SUN_STAR_DATATRANSFER_DND_DNDCONSTANS_HPP_
61 #include <com/sun/star/datatransfer/dnd/DNDConstants.hpp>
62 #endif
63 #include <com/sun/star/datatransfer/dnd/XDragGestureRecognizer.hpp>
64 #include <com/sun/star/datatransfer/dnd/XDropTarget.hpp>
65 
66 #include <vcl/edit.hxx>
67 
68 
69 #include <sot/exchange.hxx>
70 #include <sot/formats.hxx>
71 
72 #include <vos/mutex.hxx>
73 
74 
75 using namespace ::com::sun::star;
76 
77 class TETextDataObject :	public ::com::sun::star::datatransfer::XTransferable,
78 						public ::cppu::OWeakObject
79 
80 {
81 private:
82 	String			maText;
83 	SvMemoryStream	maHTMLStream;
84 
85 public:
86 					TETextDataObject( const String& rText );
87 					~TETextDataObject();
88 
89 	String&			GetText() { return maText; }
90 	SvMemoryStream&	GetHTMLStream() { return maHTMLStream; }
91 
92 	// ::com::sun::star::uno::XInterface
93     ::com::sun::star::uno::Any					SAL_CALL queryInterface( const ::com::sun::star::uno::Type & rType ) throw(::com::sun::star::uno::RuntimeException);
94 	void										SAL_CALL acquire() throw()	{ OWeakObject::acquire(); }
95 	void										SAL_CALL release() throw()	{ OWeakObject::release(); }
96 
97 	// ::com::sun::star::datatransfer::XTransferable
98     ::com::sun::star::uno::Any SAL_CALL getTransferData( const ::com::sun::star::datatransfer::DataFlavor& aFlavor ) throw(::com::sun::star::datatransfer::UnsupportedFlavorException, ::com::sun::star::io::IOException, ::com::sun::star::uno::RuntimeException);
99     ::com::sun::star::uno::Sequence< ::com::sun::star::datatransfer::DataFlavor > SAL_CALL getTransferDataFlavors(  ) throw(::com::sun::star::uno::RuntimeException);
100     sal_Bool SAL_CALL isDataFlavorSupported( const ::com::sun::star::datatransfer::DataFlavor& aFlavor ) throw(::com::sun::star::uno::RuntimeException);
101 };
102 
103 TETextDataObject::TETextDataObject( const String& rText ) : maText( rText )
104 {
105 }
106 
107 TETextDataObject::~TETextDataObject()
108 {
109 }
110 
111 // uno::XInterface
112 uno::Any TETextDataObject::queryInterface( const uno::Type & rType ) throw(uno::RuntimeException)
113 {
114 	uno::Any aRet = ::cppu::queryInterface( rType, SAL_STATIC_CAST( datatransfer::XTransferable*, this ) );
115 	return (aRet.hasValue() ? aRet : OWeakObject::queryInterface( rType ));
116 }
117 
118 // datatransfer::XTransferable
119 uno::Any TETextDataObject::getTransferData( const datatransfer::DataFlavor& rFlavor ) throw(datatransfer::UnsupportedFlavorException, io::IOException, uno::RuntimeException)
120 {
121 	uno::Any aAny;
122 
123 	sal_uLong nT = SotExchange::GetFormat( rFlavor );
124 	if ( nT == SOT_FORMAT_STRING )
125 	{
126 		aAny <<= (::rtl::OUString)GetText();
127 	}
128 	else if ( nT == SOT_FORMATSTR_ID_HTML )
129 	{
130 		GetHTMLStream().Seek( STREAM_SEEK_TO_END );
131 		sal_uLong nLen = GetHTMLStream().Tell();
132 		GetHTMLStream().Seek(0);
133 
134 		uno::Sequence< sal_Int8 > aSeq( nLen );
135 		memcpy( aSeq.getArray(), GetHTMLStream().GetData(), nLen );
136 		aAny <<= aSeq;
137 	}
138     else
139     {
140         throw datatransfer::UnsupportedFlavorException();
141     }
142 	return aAny;
143 }
144 
145 uno::Sequence< datatransfer::DataFlavor > TETextDataObject::getTransferDataFlavors(  ) throw(uno::RuntimeException)
146 {
147 	GetHTMLStream().Seek( STREAM_SEEK_TO_END );
148 	sal_Bool bHTML = GetHTMLStream().Tell() > 0;
149 	uno::Sequence< datatransfer::DataFlavor > aDataFlavors( bHTML ? 2 : 1 );
150 	SotExchange::GetFormatDataFlavor( SOT_FORMAT_STRING, aDataFlavors.getArray()[0] );
151 	if ( bHTML )
152 		SotExchange::GetFormatDataFlavor( SOT_FORMATSTR_ID_HTML, aDataFlavors.getArray()[1] );
153 	return aDataFlavors;
154 }
155 
156 sal_Bool TETextDataObject::isDataFlavorSupported( const datatransfer::DataFlavor& rFlavor ) throw(uno::RuntimeException)
157 {
158 	sal_uLong nT = SotExchange::GetFormat( rFlavor );
159 	return ( nT == SOT_FORMAT_STRING );
160 }
161 
162 /*-- 24.06.2004 13:54:36---------------------------------------------------
163 
164   -----------------------------------------------------------------------*/
165 struct ImpTextView
166 {
167     TextEngine*         mpTextEngine;
168 
169     Window*             mpWindow;
170     TextSelection       maSelection;
171     Point               maStartDocPos;
172 //    TextPaM             maMBDownPaM;
173 
174     Cursor*             mpCursor;
175 
176     TextDDInfo*         mpDDInfo;
177 
178     VirtualDevice*      mpVirtDev;
179 
180     SelectionEngine*    mpSelEngine;
181     TextSelFunctionSet* mpSelFuncSet;
182 
183     ::com::sun::star::uno::Reference< ::com::sun::star::datatransfer::dnd::XDragSourceListener > mxDnDListener;
184 
185     sal_uInt16              mnTravelXPos;
186 
187     sal_Bool                mbAutoScroll            : 1;
188     sal_Bool                mbInsertMode            : 1;
189     sal_Bool                mbReadOnly              : 1;
190     sal_Bool                mbPaintSelection        : 1;
191     sal_Bool                mbAutoIndent            : 1;
192     sal_Bool                mbHighlightSelection    : 1;
193     sal_Bool                mbCursorEnabled         : 1;
194     sal_Bool                mbClickedInSelection    : 1;
195     sal_Bool                mbSupportProtectAttribute : 1;
196     bool                mbCursorAtEndOfLine;
197 };
198 
199 // -------------------------------------------------------------------------
200 // (+) class TextView
201 // -------------------------------------------------------------------------
202 TextView::TextView( TextEngine* pEng, Window* pWindow ) :
203     mpImpl(new ImpTextView)
204 {
205     pWindow->EnableRTL( sal_False );
206 
207     mpImpl->mpWindow = pWindow;
208     mpImpl->mpTextEngine = pEng;
209     mpImpl->mpVirtDev = NULL;
210 
211     mpImpl->mbPaintSelection = sal_True;
212     mpImpl->mbAutoScroll = sal_True;
213     mpImpl->mbInsertMode = sal_True;
214     mpImpl->mbReadOnly = sal_False;
215     mpImpl->mbHighlightSelection = sal_False;
216     mpImpl->mbAutoIndent = sal_False;
217     mpImpl->mbCursorEnabled = sal_True;
218     mpImpl->mbClickedInSelection = sal_False;
219     mpImpl->mbSupportProtectAttribute = sal_False;
220     mpImpl->mbCursorAtEndOfLine = false;
221 //	mbInSelection = sal_False;
222 
223     mpImpl->mnTravelXPos = TRAVEL_X_DONTKNOW;
224 
225     mpImpl->mpSelFuncSet = new TextSelFunctionSet( this );
226     mpImpl->mpSelEngine = new SelectionEngine( mpImpl->mpWindow, mpImpl->mpSelFuncSet );
227     mpImpl->mpSelEngine->SetSelectionMode( RANGE_SELECTION );
228     mpImpl->mpSelEngine->EnableDrag( sal_True );
229 
230     mpImpl->mpCursor = new Cursor;
231     mpImpl->mpCursor->Show();
232     pWindow->SetCursor( mpImpl->mpCursor );
233     pWindow->SetInputContext( InputContext( pEng->GetFont(), INPUTCONTEXT_TEXT|INPUTCONTEXT_EXTTEXTINPUT ) );
234 
235 	if ( pWindow->GetSettings().GetStyleSettings().GetSelectionOptions() & SELECTION_OPTION_INVERT )
236         mpImpl->mbHighlightSelection = sal_True;
237 
238 	pWindow->SetLineColor();
239 
240     mpImpl->mpDDInfo = NULL;
241 
242     if ( pWindow->GetDragGestureRecognizer().is() )
243     {
244         vcl::unohelper::DragAndDropWrapper* pDnDWrapper = new vcl::unohelper::DragAndDropWrapper( this );
245         mpImpl->mxDnDListener = pDnDWrapper;
246 
247         uno::Reference< datatransfer::dnd::XDragGestureListener> xDGL( mpImpl->mxDnDListener, uno::UNO_QUERY );
248         pWindow->GetDragGestureRecognizer()->addDragGestureListener( xDGL );
249         uno::Reference< datatransfer::dnd::XDropTargetListener> xDTL( xDGL, uno::UNO_QUERY );
250         pWindow->GetDropTarget()->addDropTargetListener( xDTL );
251         pWindow->GetDropTarget()->setActive( sal_True );
252         pWindow->GetDropTarget()->setDefaultActions( datatransfer::dnd::DNDConstants::ACTION_COPY_OR_MOVE );
253     }
254 }
255 
256 TextView::~TextView()
257 {
258     delete mpImpl->mpSelEngine;
259     delete mpImpl->mpSelFuncSet;
260     delete mpImpl->mpVirtDev;
261 
262     if ( mpImpl->mpWindow->GetCursor() == mpImpl->mpCursor )
263         mpImpl->mpWindow->SetCursor( 0 );
264     delete mpImpl->mpCursor;
265     delete mpImpl->mpDDInfo;
266     delete mpImpl;
267 }
268 
269 void TextView::Invalidate()
270 {
271     mpImpl->mpWindow->Invalidate();
272 }
273 
274 void TextView::SetSelection( const TextSelection& rTextSel, sal_Bool bGotoCursor )
275 {
276 	// Falls jemand gerade ein leeres Attribut hinterlassen hat,
277 	// und dann der Outliner die Selektion manipulitert:
278     if ( !mpImpl->maSelection.HasRange() )
279         mpImpl->mpTextEngine->CursorMoved( mpImpl->maSelection.GetStart().GetPara() );
280 
281 	// Wenn nach einem KeyInput die Selection manipuliert wird:
282     mpImpl->mpTextEngine->CheckIdleFormatter();
283 
284 	HideSelection();
285     TextSelection aNewSel( rTextSel );
286     mpImpl->mpTextEngine->ValidateSelection( aNewSel );
287 	ImpSetSelection( aNewSel );
288 	ShowSelection();
289 	ShowCursor( bGotoCursor );
290 }
291 
292 void TextView::SetSelection( const TextSelection& rTextSel )
293 {
294     SetSelection( rTextSel, mpImpl->mbAutoScroll );
295 }
296 
297 const TextSelection& TextView::GetSelection() const
298 {
299     return mpImpl->maSelection;
300 }
301 TextSelection&      TextView::GetSelection()
302 {
303     return mpImpl->maSelection;
304 }
305 
306 void TextView::DeleteSelected()
307 {
308 //	HideSelection();
309 
310     mpImpl->mpTextEngine->UndoActionStart();
311     TextPaM aPaM = mpImpl->mpTextEngine->ImpDeleteText( mpImpl->maSelection );
312     mpImpl->mpTextEngine->UndoActionEnd();
313 
314 	ImpSetSelection( aPaM );
315     mpImpl->mpTextEngine->FormatAndUpdate( this );
316 	ShowCursor();
317 }
318 
319 void TextView::ImpPaint( OutputDevice* pOut, const Point& rStartPos, Rectangle const* pPaintArea, TextSelection const* pPaintRange, TextSelection const* pSelection )
320 {
321     if ( !mpImpl->mbPaintSelection )
322 		pSelection = NULL;
323 	else
324 	{
325 		// Richtige Hintergrundfarbe einstellen.
326 		// Ich bekomme leider nicht mit, ob sich diese inzwischen geaendert hat.
327         Font aFont = mpImpl->mpTextEngine->GetFont();
328 		Color aColor = pOut->GetBackground().GetColor();
329 		aColor.SetTransparency( 0 );
330 		if ( aColor != aFont.GetFillColor() )
331 		{
332             if( aFont.IsTransparent() )
333                 aColor = Color( COL_TRANSPARENT );
334 			aFont.SetFillColor( aColor );
335             mpImpl->mpTextEngine->maFont = aFont;
336 		}
337 	}
338 
339     mpImpl->mpTextEngine->ImpPaint( pOut, rStartPos, pPaintArea, pPaintRange, pSelection );
340 }
341 
342 void TextView::Paint( const Rectangle& rRect )
343 {
344 	ImpPaint( rRect, sal_False );
345 }
346 
347 void TextView::ImpPaint( const Rectangle& rRect, sal_Bool bUseVirtDev )
348 {
349     if ( !mpImpl->mpTextEngine->GetUpdateMode() || mpImpl->mpTextEngine->IsInUndo() )
350 		return;
351 
352 	TextSelection *pDrawSelection = NULL;
353     if ( !mpImpl->mbHighlightSelection && mpImpl->maSelection.HasRange() )
354         pDrawSelection = &mpImpl->maSelection;
355 
356 	if ( bUseVirtDev )
357 	{
358 		VirtualDevice* pVDev = GetVirtualDevice();
359 
360         const Color& rBackgroundColor = mpImpl->mpWindow->GetBackground().GetColor();
361 		if ( pVDev->GetFillColor() != rBackgroundColor )
362 			pVDev->SetFillColor( rBackgroundColor );
363 		if ( pVDev->GetBackground().GetColor() != rBackgroundColor )
364 			pVDev->SetBackground( rBackgroundColor );
365 
366 		sal_Bool bVDevValid = sal_True;
367 		Size aOutSz( pVDev->GetOutputSizePixel() );
368 		if ( (	aOutSz.Width() < rRect.GetWidth() ) ||
369 			 (	aOutSz.Height() < rRect.GetHeight() ) )
370 		{
371 			bVDevValid = pVDev->SetOutputSizePixel( rRect.GetSize() );
372 		}
373 		else
374 		{
375 			// Das VirtDev kann bei einem Resize sehr gross werden =>
376 			// irgendwann mal kleiner machen!
377 			if ( ( aOutSz.Height() > ( rRect.GetHeight() + 20 ) ) ||
378 				 ( aOutSz.Width() > ( rRect.GetWidth() + 20 ) ) )
379 			{
380 				bVDevValid = pVDev->SetOutputSizePixel( rRect.GetSize() );
381 			}
382 			else
383 			{
384 				pVDev->Erase();
385 			}
386 		}
387 		if ( !bVDevValid )
388 		{
389 			ImpPaint( rRect, sal_False /* ohne VDev */ );
390 			return;
391 		}
392 
393 		Rectangle aTmpRec( Point( 0, 0 ), rRect.GetSize() );
394 
395         Point aDocPos( mpImpl->maStartDocPos.X(), mpImpl->maStartDocPos.Y() + rRect.Top() );
396         Point aStartPos = ImpGetOutputStartPos( aDocPos );
397 		ImpPaint( pVDev, aStartPos, &aTmpRec, NULL, pDrawSelection );
398         mpImpl->mpWindow->DrawOutDev( rRect.TopLeft(), rRect.GetSize(),
399 								Point(0,0), rRect.GetSize(), *pVDev );
400 //		ShowSelection();
401         if ( mpImpl->mbHighlightSelection )
402             ImpHighlight( mpImpl->maSelection );
403 	}
404 	else
405 	{
406         Point aStartPos = ImpGetOutputStartPos( mpImpl->maStartDocPos );
407         ImpPaint( mpImpl->mpWindow, aStartPos, &rRect, NULL, pDrawSelection );
408 
409 //		ShowSelection();
410         if ( mpImpl->mbHighlightSelection )
411             ImpHighlight( mpImpl->maSelection );
412 	}
413 }
414 
415 void TextView::ImpHighlight( const TextSelection& rSel )
416 {
417 	TextSelection aSel( rSel );
418 	aSel.Justify();
419     if ( aSel.HasRange() && !mpImpl->mpTextEngine->IsInUndo() && mpImpl->mpTextEngine->GetUpdateMode() )
420 	{
421         mpImpl->mpCursor->Hide();
422 
423         DBG_ASSERT( !mpImpl->mpTextEngine->mpIdleFormatter->IsActive(), "ImpHighlight: Not formatted!" );
424 
425         Rectangle aVisArea( mpImpl->maStartDocPos, mpImpl->mpWindow->GetOutputSizePixel() );
426 		long nY = 0;
427 		sal_uLong nStartPara = aSel.GetStart().GetPara();
428 		sal_uLong nEndPara = aSel.GetEnd().GetPara();
429 		for ( sal_uLong nPara = 0; nPara <= nEndPara; nPara++ )
430 		{
431             long nParaHeight = (long)mpImpl->mpTextEngine->CalcParaHeight( nPara );
432 			if ( ( nPara >= nStartPara ) && ( ( nY + nParaHeight ) > aVisArea.Top() ) )
433 			{
434                 TEParaPortion* pTEParaPortion = mpImpl->mpTextEngine->mpTEParaPortions->GetObject( nPara );
435 				sal_uInt16 nStartLine = 0;
436 				sal_uInt16 nEndLine = pTEParaPortion->GetLines().Count() -1;
437 				if ( nPara == nStartPara )
438 					nStartLine = pTEParaPortion->GetLineNumber( aSel.GetStart().GetIndex(), sal_False );
439 				if ( nPara == nEndPara )
440 					nEndLine = pTEParaPortion->GetLineNumber( aSel.GetEnd().GetIndex(), sal_True );
441 
442 				// ueber die Zeilen iterieren....
443 				for ( sal_uInt16 nLine = nStartLine; nLine <= nEndLine; nLine++ )
444 				{
445 					TextLine* pLine = pTEParaPortion->GetLines().GetObject( nLine );
446 					sal_uInt16 nStartIndex = pLine->GetStart();
447 					sal_uInt16 nEndIndex = pLine->GetEnd();
448 					if ( ( nPara == nStartPara ) && ( nLine == nStartLine ) )
449 						nStartIndex = aSel.GetStart().GetIndex();
450 					if ( ( nPara == nEndPara ) && ( nLine == nEndLine ) )
451 						nEndIndex = aSel.GetEnd().GetIndex();
452 
453 					// Kann passieren, wenn am Anfang einer umgebrochenen Zeile.
454 					if ( nEndIndex < nStartIndex )
455 						nEndIndex = nStartIndex;
456 
457                     Rectangle aTmpRec( mpImpl->mpTextEngine->GetEditCursor( TextPaM( nPara, nStartIndex ), sal_False ) );
458 					aTmpRec.Top() += nY;
459 					aTmpRec.Bottom() += nY;
460 					Point aTopLeft( aTmpRec.TopLeft() );
461 
462                     aTmpRec = mpImpl->mpTextEngine->GetEditCursor( TextPaM( nPara, nEndIndex ), sal_True );
463 					aTmpRec.Top() += nY;
464 					aTmpRec.Bottom() += nY;
465 					Point aBottomRight( aTmpRec.BottomRight() );
466 					aBottomRight.X()--;
467 
468 					// Nur Painten, wenn im sichtbaren Bereich...
469 					if ( ( aTopLeft.X() < aBottomRight.X() ) && ( aBottomRight.Y() >= aVisArea.Top() ) )
470 					{
471 						Point aPnt1( GetWindowPos( aTopLeft ) );
472 						Point aPnt2( GetWindowPos( aBottomRight ) );
473 
474 						Rectangle aRect( aPnt1, aPnt2 );
475                         mpImpl->mpWindow->Invert( aRect );
476 					}
477 				}
478 			}
479 			nY += nParaHeight;
480 
481 			if ( nY >= aVisArea.Bottom() )
482 				break;
483 		}
484 	}
485 }
486 
487 void TextView::ImpSetSelection( const TextSelection& rSelection )
488 {
489     if ( rSelection != mpImpl->maSelection )
490     {
491         mpImpl->maSelection = rSelection;
492         mpImpl->mpTextEngine->Broadcast( TextHint( TEXT_HINT_VIEWSELECTIONCHANGED ) );
493     }
494 }
495 
496 void TextView::ShowSelection()
497 {
498     ImpShowHideSelection( sal_True );
499 }
500 
501 void TextView::HideSelection()
502 {
503     ImpShowHideSelection( sal_False );
504 }
505 
506 void TextView::ShowSelection( const TextSelection& rRange )
507 {
508     ImpShowHideSelection( sal_True, &rRange );
509 }
510 
511 void TextView::ImpShowHideSelection( sal_Bool bShow, const TextSelection* pRange )
512 {
513     const TextSelection* pRangeOrSelection = pRange ? pRange : &mpImpl->maSelection;
514 
515 	if ( pRangeOrSelection->HasRange() )
516 	{
517         if ( mpImpl->mbHighlightSelection )
518         {
519 			ImpHighlight( *pRangeOrSelection );
520         }
521 		else
522 		{
523             if( mpImpl->mpWindow->IsPaintTransparent() )
524                 mpImpl->mpWindow->Invalidate();
525             else
526             {
527                 Rectangle aOutArea( Point( 0, 0 ), mpImpl->mpWindow->GetOutputSizePixel() );
528                 Point aStartPos( ImpGetOutputStartPos( mpImpl->maStartDocPos ) );
529                 TextSelection aRange( *pRangeOrSelection );
530                 aRange.Justify();
531                 sal_Bool bVisCursor = mpImpl->mpCursor->IsVisible();
532                 mpImpl->mpCursor->Hide();
533                 ImpPaint( mpImpl->mpWindow, aStartPos, &aOutArea, &aRange, bShow ? &mpImpl->maSelection : NULL );
534                 if ( bVisCursor )
535                     mpImpl->mpCursor->Show();
536             }
537 		}
538 	}
539 }
540 
541 VirtualDevice* TextView::GetVirtualDevice()
542 {
543     if ( !mpImpl->mpVirtDev )
544 	{
545         mpImpl->mpVirtDev = new VirtualDevice;
546         mpImpl->mpVirtDev->SetLineColor();
547 	}
548     return mpImpl->mpVirtDev;
549 }
550 
551 void TextView::EraseVirtualDevice()
552 {
553     delete mpImpl->mpVirtDev;
554     mpImpl->mpVirtDev = 0;
555 }
556 
557 sal_Bool TextView::KeyInput( const KeyEvent& rKeyEvent )
558 {
559 	sal_Bool bDone 		= sal_True;
560 	sal_Bool bModified	= sal_False;
561 	sal_Bool bMoved		= sal_False;
562 	sal_Bool bEndKey	= sal_False;	// spezielle CursorPosition
563 	sal_Bool bAllowIdle = sal_True;
564 
565 	// Um zu pruefen ob durch irgendeine Aktion mModified, das lokale
566 	// bModified wird z.B. bei Cut/Paste nicht gesetzt, weil dort an anderen
567 	// Stellen das updaten erfolgt.
568     sal_Bool bWasModified = mpImpl->mpTextEngine->IsModified();
569     mpImpl->mpTextEngine->SetModified( sal_False );
570 
571     TextSelection aCurSel( mpImpl->maSelection );
572 	TextSelection aOldSel( aCurSel );
573 
574 	sal_uInt16 nCode = rKeyEvent.GetKeyCode().GetCode();
575 	KeyFuncType eFunc = rKeyEvent.GetKeyCode().GetFunction();
576 	if ( eFunc != KEYFUNC_DONTKNOW )
577 	{
578 		switch ( eFunc )
579 		{
580 			case KEYFUNC_CUT:
581 			{
582                 if ( !mpImpl->mbReadOnly )
583 					Cut();
584 			}
585 			break;
586 			case KEYFUNC_COPY:
587 			{
588 				Copy();
589 			}
590 			break;
591 			case KEYFUNC_PASTE:
592 			{
593                 if ( !mpImpl->mbReadOnly )
594 					Paste();
595 			}
596 			break;
597 			case KEYFUNC_UNDO:
598 			{
599                 if ( !mpImpl->mbReadOnly )
600 					Undo();
601 			}
602 			break;
603 			case KEYFUNC_REDO:
604 			{
605                 if ( !mpImpl->mbReadOnly )
606 					Redo();
607 			}
608 			break;
609 
610 			default:	// wird dann evtl. unten bearbeitet.
611 						eFunc = KEYFUNC_DONTKNOW;
612 		}
613 	}
614 	if ( eFunc == KEYFUNC_DONTKNOW )
615 	{
616 		switch ( nCode )
617 		{
618 			case KEY_UP:
619 			case KEY_DOWN:
620 			case KEY_LEFT:
621 			case KEY_RIGHT:
622 			case KEY_HOME:
623 			case KEY_END:
624 			case KEY_PAGEUP:
625 			case KEY_PAGEDOWN:
626             case com::sun::star::awt::Key::MOVE_WORD_FORWARD:
627             case com::sun::star::awt::Key::SELECT_WORD_FORWARD:
628             case com::sun::star::awt::Key::MOVE_WORD_BACKWARD:
629             case com::sun::star::awt::Key::SELECT_WORD_BACKWARD:
630             case com::sun::star::awt::Key::MOVE_TO_BEGIN_OF_LINE:
631             case com::sun::star::awt::Key::MOVE_TO_END_OF_LINE:
632             case com::sun::star::awt::Key::SELECT_TO_BEGIN_OF_LINE:
633             case com::sun::star::awt::Key::SELECT_TO_END_OF_LINE:
634             case com::sun::star::awt::Key::MOVE_TO_BEGIN_OF_PARAGRAPH:
635             case com::sun::star::awt::Key::MOVE_TO_END_OF_PARAGRAPH:
636             case com::sun::star::awt::Key::SELECT_TO_BEGIN_OF_PARAGRAPH:
637             case com::sun::star::awt::Key::SELECT_TO_END_OF_PARAGRAPH:
638             case com::sun::star::awt::Key::MOVE_TO_BEGIN_OF_DOCUMENT:
639             case com::sun::star::awt::Key::MOVE_TO_END_OF_DOCUMENT:
640             case com::sun::star::awt::Key::SELECT_TO_BEGIN_OF_DOCUMENT:
641             case com::sun::star::awt::Key::SELECT_TO_END_OF_DOCUMENT:
642 			{
643 				if ( ( !rKeyEvent.GetKeyCode().IsMod2() || ( nCode == KEY_LEFT ) || ( nCode == KEY_RIGHT ) )
644                       && !( rKeyEvent.GetKeyCode().IsMod1() && ( nCode == KEY_PAGEUP || nCode == KEY_PAGEDOWN ) ) )
645 				{
646 					aCurSel = ImpMoveCursor( rKeyEvent );
647                     if ( aCurSel.HasRange() ) {
648                         uno::Reference<datatransfer::clipboard::XClipboard> aSelection(GetWindow()->GetPrimarySelection());
649                         Copy( aSelection );
650                     }
651 					bMoved = sal_True;
652 					if ( nCode == KEY_END )
653 						bEndKey = sal_True;
654 				}
655 				else
656 					bDone = sal_False;
657 			}
658 			break;
659 			case KEY_BACKSPACE:
660 			case KEY_DELETE:
661             case com::sun::star::awt::Key::DELETE_WORD_BACKWARD:
662             case com::sun::star::awt::Key::DELETE_WORD_FORWARD:
663             case com::sun::star::awt::Key::DELETE_TO_BEGIN_OF_LINE:
664             case com::sun::star::awt::Key::DELETE_TO_END_OF_LINE:
665 			{
666                 if ( !mpImpl->mbReadOnly && !rKeyEvent.GetKeyCode().IsMod2() )
667 				{
668 					sal_uInt8 nDel = ( nCode == KEY_DELETE ) ? DEL_RIGHT : DEL_LEFT;
669 					sal_uInt8 nMode = rKeyEvent.GetKeyCode().IsMod1() ? DELMODE_RESTOFWORD : DELMODE_SIMPLE;
670 					if ( ( nMode == DELMODE_RESTOFWORD ) && rKeyEvent.GetKeyCode().IsShift() )
671 						nMode = DELMODE_RESTOFCONTENT;
672 
673                     switch( nCode )
674                     {
675                     case com::sun::star::awt::Key::DELETE_WORD_BACKWARD:
676                         nDel = DEL_LEFT;
677                         nMode = DELMODE_RESTOFWORD;
678                         break;
679                     case com::sun::star::awt::Key::DELETE_WORD_FORWARD:
680                         nDel = DEL_RIGHT;
681                         nMode = DELMODE_RESTOFWORD;
682                         break;
683                     case com::sun::star::awt::Key::DELETE_TO_BEGIN_OF_LINE:
684                         nDel = DEL_LEFT;
685                         nMode = DELMODE_RESTOFCONTENT;
686                         break;
687                     case com::sun::star::awt::Key::DELETE_TO_END_OF_LINE:
688                         nDel = DEL_RIGHT;
689                         nMode = DELMODE_RESTOFCONTENT;
690                         break;
691                     default: break;
692                     }
693 
694                     mpImpl->mpTextEngine->UndoActionStart();
695                     if(mpImpl->mbSupportProtectAttribute)
696                     {
697                         //expand selection to include all protected content - if there is any
698                         const TextCharAttrib* pStartAttr = mpImpl->mpTextEngine->FindCharAttrib(
699                                     TextPaM(mpImpl->maSelection.GetStart().GetPara(),
700                                     mpImpl->maSelection.GetStart().GetIndex()),
701                                     TEXTATTR_PROTECTED );
702                         const TextCharAttrib* pEndAttr = mpImpl->mpTextEngine->FindCharAttrib(
703                                     TextPaM(mpImpl->maSelection.GetEnd().GetPara(),
704                                     mpImpl->maSelection.GetEnd().GetIndex()),
705                                     TEXTATTR_PROTECTED );
706                         if(pStartAttr && pStartAttr->GetStart() < mpImpl->maSelection.GetStart().GetIndex())
707                         {
708                             mpImpl->maSelection.GetStart().GetIndex() = pStartAttr->GetStart();
709                         }
710                         if(pEndAttr && pEndAttr->GetEnd() > mpImpl->maSelection.GetEnd().GetIndex())
711                         {
712                             mpImpl->maSelection.GetEnd().GetIndex() = pEndAttr->GetEnd();
713                         }
714                     }
715                     aCurSel = ImpDelete( nDel, nMode );
716                     mpImpl->mpTextEngine->UndoActionEnd();
717 					bModified = sal_True;
718 					bAllowIdle = sal_False;
719 				}
720 				else
721 					bDone = sal_False;
722 			}
723 			break;
724 			case KEY_TAB:
725 			{
726                 if ( !mpImpl->mbReadOnly && !rKeyEvent.GetKeyCode().IsShift() &&
727 						!rKeyEvent.GetKeyCode().IsMod1() && !rKeyEvent.GetKeyCode().IsMod2() &&
728 						ImplCheckTextLen( 'x' ) )
729 				{
730                     aCurSel = mpImpl->mpTextEngine->ImpInsertText( aCurSel, '\t', !IsInsertMode() );
731 					bModified = sal_True;
732 				}
733 				else
734 					bDone = sal_False;
735 			}
736 			break;
737 			case KEY_RETURN:
738 			{
739 				// Shift-RETURN darf nicht geschluckt werden, weil dann keine
740 				// mehrzeilige Eingabe in Dialogen/Property-Editor moeglich.
741                 if ( !mpImpl->mbReadOnly && !rKeyEvent.GetKeyCode().IsMod1() &&
742 						!rKeyEvent.GetKeyCode().IsMod2() && ImplCheckTextLen( 'x' ) )
743 				{
744                     mpImpl->mpTextEngine->UndoActionStart();
745                     aCurSel = mpImpl->mpTextEngine->ImpInsertParaBreak( aCurSel );
746                     if ( mpImpl->mbAutoIndent )
747 					{
748                         TextNode* pPrev = mpImpl->mpTextEngine->mpDoc->GetNodes().GetObject( aCurSel.GetEnd().GetPara() - 1 );
749 						sal_uInt16 n = 0;
750 						while ( ( n < pPrev->GetText().Len() ) && (
751 									( pPrev->GetText().GetChar( n ) == ' ' ) ||
752 									( pPrev->GetText().GetChar( n ) == '\t' ) ) )
753 						{
754 							n++;
755 						}
756 						if ( n )
757                             aCurSel = mpImpl->mpTextEngine->ImpInsertText( aCurSel, pPrev->GetText().Copy( 0, n ) );
758 					}
759                     mpImpl->mpTextEngine->UndoActionEnd();
760 					bModified = sal_True;
761 				}
762 				else
763 					bDone = sal_False;
764 			}
765 			break;
766 			case KEY_INSERT:
767 			{
768                 if ( !mpImpl->mbReadOnly )
769 					SetInsertMode( !IsInsertMode() );
770 			}
771 			break;
772 			default:
773 			{
774 				if ( TextEngine::IsSimpleCharInput( rKeyEvent ) )
775 				{
776 					xub_Unicode nCharCode = rKeyEvent.GetCharCode();
777                     if ( !mpImpl->mbReadOnly && ImplCheckTextLen( nCharCode ) )    // sonst trotzdem das Zeichen schlucken...
778 					{
779                         aCurSel = mpImpl->mpTextEngine->ImpInsertText( nCharCode, aCurSel, !IsInsertMode(), sal_True );
780 						bModified = sal_True;
781 					}
782 				}
783 				else
784 					bDone = sal_False;
785 			}
786 		}
787 	}
788 
789     if ( aCurSel != aOldSel )   // Check if changed, maybe other method already changed mpImpl->maSelection, don't overwrite that!
790         ImpSetSelection( aCurSel );
791 
792     mpImpl->mpTextEngine->UpdateSelections();
793 
794 	if ( ( nCode != KEY_UP ) && ( nCode != KEY_DOWN ) )
795         mpImpl->mnTravelXPos = TRAVEL_X_DONTKNOW;
796 
797 	if ( bModified )
798 	{
799 		// Idle-Formatter nur, wenn AnyInput.
800 		if ( bAllowIdle && Application::AnyInput( INPUT_KEYBOARD) )
801             mpImpl->mpTextEngine->IdleFormatAndUpdate( this );
802 		else
803             mpImpl->mpTextEngine->FormatAndUpdate( this);
804 	}
805 	else if ( bMoved )
806 	{
807 		// Selection wird jetzt gezielt in ImpMoveCursor gemalt.
808         ImpShowCursor( mpImpl->mbAutoScroll, sal_True, bEndKey );
809 	}
810 
811     if ( mpImpl->mpTextEngine->IsModified() )
812         mpImpl->mpTextEngine->Broadcast( TextHint( TEXT_HINT_MODIFIED ) );
813 	else if ( bWasModified )
814         mpImpl->mpTextEngine->SetModified( sal_True );
815 
816 	return bDone;
817 }
818 
819 void TextView::MouseButtonUp( const MouseEvent& rMouseEvent )
820 {
821     mpImpl->mbClickedInSelection = sal_False;
822     mpImpl->mnTravelXPos = TRAVEL_X_DONTKNOW;
823     mpImpl->mpSelEngine->SelMouseButtonUp( rMouseEvent );
824     if ( rMouseEvent.IsMiddle() && !IsReadOnly() &&
825          ( GetWindow()->GetSettings().GetMouseSettings().GetMiddleButtonAction() == MOUSE_MIDDLE_PASTESELECTION ) )
826     {
827         uno::Reference<datatransfer::clipboard::XClipboard> aSelection(GetWindow()->GetPrimarySelection());
828         Paste( aSelection );
829         if ( mpImpl->mpTextEngine->IsModified() )
830             mpImpl->mpTextEngine->Broadcast( TextHint( TEXT_HINT_MODIFIED ) );
831     }
832     else if ( rMouseEvent.IsLeft() && GetSelection().HasRange() )
833     {
834         uno::Reference<datatransfer::clipboard::XClipboard> aSelection(GetWindow()->GetPrimarySelection());
835         Copy( aSelection );
836     }
837 }
838 
839 void TextView::MouseButtonDown( const MouseEvent& rMouseEvent )
840 {
841     mpImpl->mpTextEngine->CheckIdleFormatter();    // Falls schnelles Tippen und MouseButtonDown
842     mpImpl->mnTravelXPos = TRAVEL_X_DONTKNOW;
843     mpImpl->mbClickedInSelection = IsSelectionAtPoint( rMouseEvent.GetPosPixel() );
844 
845     mpImpl->mpTextEngine->SetActiveView( this );
846 
847     mpImpl->mpSelEngine->SelMouseButtonDown( rMouseEvent );
848 
849 	// mbu 20.01.2005 - SelMouseButtonDown() possibly triggers a 'selection changed'
850 	// notification. The appropriate handler could change the current selection,
851 	// which is the case in the MailMerge address block control. To enable select'n'drag
852 	// we need to reevaluate the selection after the notification has been fired.
853     mpImpl->mbClickedInSelection = IsSelectionAtPoint( rMouseEvent.GetPosPixel() );
854 
855 	// Sonderbehandlungen
856 	if ( !rMouseEvent.IsShift() && ( rMouseEvent.GetClicks() >= 2 ) )
857 	{
858 		if ( rMouseEvent.IsMod2() )
859 		{
860 			HideSelection();
861             ImpSetSelection( mpImpl->maSelection.GetEnd() );
862 			SetCursorAtPoint( rMouseEvent.GetPosPixel() );	// Wird von SelectionEngine bei MOD2 nicht gesetzt
863 		}
864 
865 		if ( rMouseEvent.GetClicks() == 2 )
866 		{
867 			// Wort selektieren
868             if ( mpImpl->maSelection.GetEnd().GetIndex() < mpImpl->mpTextEngine->GetTextLen( mpImpl->maSelection.GetEnd().GetPara() ) )
869 			{
870 				HideSelection();
871                 TextNode* pNode = mpImpl->mpTextEngine->mpDoc->GetNodes().GetObject(  mpImpl->maSelection.GetEnd().GetPara() );
872                 uno::Reference < i18n::XBreakIterator > xBI = mpImpl->mpTextEngine->GetBreakIterator();
873                 i18n::Boundary aBoundary = xBI->getWordBoundary( pNode->GetText(), mpImpl->maSelection.GetEnd().GetIndex(), mpImpl->mpTextEngine->GetLocale(), i18n::WordType::ANYWORD_IGNOREWHITESPACES, sal_True );
874                 TextSelection aNewSel( mpImpl->maSelection );
875 				aNewSel.GetStart().GetIndex() = (sal_uInt16)aBoundary.startPos;
876 				aNewSel.GetEnd().GetIndex() = (sal_uInt16)aBoundary.endPos;
877                 if(mpImpl->mbSupportProtectAttribute)
878                 {
879                     //expand selection to include all protected content - if there is any
880                     const TextCharAttrib* pStartAttr = mpImpl->mpTextEngine->FindCharAttrib(
881                                 TextPaM(aNewSel.GetStart().GetPara(),
882                                 (sal_uInt16)aBoundary.startPos),
883                                 TEXTATTR_PROTECTED );
884                     const TextCharAttrib* pEndAttr = mpImpl->mpTextEngine->FindCharAttrib(
885                                 TextPaM(aNewSel.GetEnd().GetPara(),
886                                 (sal_uInt16)aBoundary.endPos),
887                                 TEXTATTR_PROTECTED );
888                     if(pStartAttr && pStartAttr->GetStart() < aNewSel.GetStart().GetIndex())
889                     {
890                         aNewSel.GetStart().GetIndex() = pStartAttr->GetStart();
891                     }
892                     if(pEndAttr && pEndAttr->GetEnd() > aNewSel.GetEnd().GetIndex())
893                     {
894                         aNewSel.GetEnd().GetIndex() = pEndAttr->GetEnd();
895                     }
896                 }
897                 ImpSetSelection( aNewSel );
898 				ShowSelection();
899 				ShowCursor( sal_True, sal_True );
900 			}
901 		}
902 		else if ( rMouseEvent.GetClicks() == 3 )
903 		{
904 			// Absatz selektieren
905             if ( mpImpl->maSelection.GetStart().GetIndex() || ( mpImpl->maSelection.GetEnd().GetIndex() < mpImpl->mpTextEngine->GetTextLen( mpImpl->maSelection.GetEnd().GetPara() ) ) )
906 			{
907 				HideSelection();
908                 TextSelection aNewSel( mpImpl->maSelection );
909 				aNewSel.GetStart().GetIndex() = 0;
910                 aNewSel.GetEnd().GetIndex() = mpImpl->mpTextEngine->mpDoc->GetNodes().GetObject( mpImpl->maSelection.GetEnd().GetPara() )->GetText().Len();
911                 ImpSetSelection( aNewSel );
912 				ShowSelection();
913 				ShowCursor( sal_True, sal_True );
914 			}
915 		}
916 	}
917 }
918 
919 
920 void TextView::MouseMove( const MouseEvent& rMouseEvent )
921 {
922     mpImpl->mnTravelXPos = TRAVEL_X_DONTKNOW;
923     mpImpl->mpSelEngine->SelMouseMove( rMouseEvent );
924 }
925 
926 void TextView::Command( const CommandEvent& rCEvt )
927 {
928     mpImpl->mpTextEngine->CheckIdleFormatter();    // Falls schnelles Tippen und MouseButtonDown
929     mpImpl->mpTextEngine->SetActiveView( this );
930 
931 	if ( rCEvt.GetCommand() == COMMAND_STARTEXTTEXTINPUT )
932 	{
933 		DeleteSelected();
934         delete mpImpl->mpTextEngine->mpIMEInfos;
935         TextNode* pNode = mpImpl->mpTextEngine->mpDoc->GetNodes().GetObject( GetSelection().GetEnd().GetPara() );
936         mpImpl->mpTextEngine->mpIMEInfos = new TEIMEInfos( GetSelection().GetEnd(), pNode->GetText().Copy( GetSelection().GetEnd().GetIndex() ) );
937         mpImpl->mpTextEngine->mpIMEInfos->bWasCursorOverwrite = !IsInsertMode();
938 	}
939 	else if ( rCEvt.GetCommand() == COMMAND_ENDEXTTEXTINPUT )
940 	{
941         DBG_ASSERT( mpImpl->mpTextEngine->mpIMEInfos, "COMMAND_ENDEXTTEXTINPUT => Kein Start ?" );
942         if( mpImpl->mpTextEngine->mpIMEInfos )
943 		{
944             TEParaPortion* pPortion = mpImpl->mpTextEngine->mpTEParaPortions->GetObject( mpImpl->mpTextEngine->mpIMEInfos->aPos.GetPara() );
945             pPortion->MarkSelectionInvalid( mpImpl->mpTextEngine->mpIMEInfos->aPos.GetIndex(), 0 );
946 
947             sal_Bool bInsertMode = !mpImpl->mpTextEngine->mpIMEInfos->bWasCursorOverwrite;
948 
949             delete mpImpl->mpTextEngine->mpIMEInfos;
950             mpImpl->mpTextEngine->mpIMEInfos = NULL;
951 
952             mpImpl->mpTextEngine->FormatAndUpdate( this );
953 
954 			SetInsertMode( bInsertMode );
955 
956             if ( mpImpl->mpTextEngine->IsModified() )
957                 mpImpl->mpTextEngine->Broadcast( TextHint( TEXT_HINT_MODIFIED ) );
958 		}
959 	}
960 	else if ( rCEvt.GetCommand() == COMMAND_EXTTEXTINPUT )
961 	{
962         DBG_ASSERT( mpImpl->mpTextEngine->mpIMEInfos, "COMMAND_EXTTEXTINPUT => Kein Start ?" );
963         if( mpImpl->mpTextEngine->mpIMEInfos )
964 		{
965 			const CommandExtTextInputData* pData = rCEvt.GetExtTextInputData();
966 
967 			if ( !pData->IsOnlyCursorChanged() )
968 			{
969                 TextSelection aSelect( mpImpl->mpTextEngine->mpIMEInfos->aPos );
970                 aSelect.GetEnd().GetIndex() = aSelect.GetEnd().GetIndex() + mpImpl->mpTextEngine->mpIMEInfos->nLen;
971                 aSelect = mpImpl->mpTextEngine->ImpDeleteText( aSelect );
972                 aSelect = mpImpl->mpTextEngine->ImpInsertText( aSelect, pData->GetText() );
973 
974                 if ( mpImpl->mpTextEngine->mpIMEInfos->bWasCursorOverwrite )
975                 {
976                     sal_uInt16 nOldIMETextLen = mpImpl->mpTextEngine->mpIMEInfos->nLen;
977                     sal_uInt16 nNewIMETextLen = pData->GetText().Len();
978 
979                     if ( ( nOldIMETextLen > nNewIMETextLen ) &&
980                          ( nNewIMETextLen < mpImpl->mpTextEngine->mpIMEInfos->aOldTextAfterStartPos.Len() ) )
981                     {
982                         // restore old characters
983                         sal_uInt16 nRestore = nOldIMETextLen - nNewIMETextLen;
984                         TextPaM aPaM( mpImpl->mpTextEngine->mpIMEInfos->aPos );
985                         aPaM.GetIndex() = aPaM.GetIndex() + nNewIMETextLen;
986                         mpImpl->mpTextEngine->ImpInsertText( aPaM, mpImpl->mpTextEngine->mpIMEInfos->aOldTextAfterStartPos.Copy( nNewIMETextLen, nRestore ) );
987                     }
988                     else if ( ( nOldIMETextLen < nNewIMETextLen ) &&
989                               ( nOldIMETextLen < mpImpl->mpTextEngine->mpIMEInfos->aOldTextAfterStartPos.Len() ) )
990                     {
991                         // overwrite
992                         sal_uInt16 nOverwrite = nNewIMETextLen - nOldIMETextLen;
993                         if ( ( nOldIMETextLen + nOverwrite ) > mpImpl->mpTextEngine->mpIMEInfos->aOldTextAfterStartPos.Len() )
994                             nOverwrite = mpImpl->mpTextEngine->mpIMEInfos->aOldTextAfterStartPos.Len() - nOldIMETextLen;
995                         DBG_ASSERT( nOverwrite && (nOverwrite < 0xFF00), "IME Overwrite?!" );
996                         TextPaM aPaM( mpImpl->mpTextEngine->mpIMEInfos->aPos );
997                         aPaM.GetIndex() = aPaM.GetIndex() + nNewIMETextLen;
998                         TextSelection aSel( aPaM );
999                         aSel.GetEnd().GetIndex() =
1000                             aSel.GetEnd().GetIndex() + nOverwrite;
1001                         mpImpl->mpTextEngine->ImpDeleteText( aSel );
1002                     }
1003                 }
1004 
1005 			    if ( pData->GetTextAttr() )
1006 			    {
1007                     mpImpl->mpTextEngine->mpIMEInfos->CopyAttribs( pData->GetTextAttr(), pData->GetText().Len() );
1008                     mpImpl->mpTextEngine->mpIMEInfos->bCursor = pData->IsCursorVisible();
1009 			    }
1010 			    else
1011 			    {
1012                     mpImpl->mpTextEngine->mpIMEInfos->DestroyAttribs();
1013 			    }
1014 
1015                 TEParaPortion* pPPortion = mpImpl->mpTextEngine->mpTEParaPortions->GetObject( mpImpl->mpTextEngine->mpIMEInfos->aPos.GetPara() );
1016                 pPPortion->MarkSelectionInvalid( mpImpl->mpTextEngine->mpIMEInfos->aPos.GetIndex(), 0 );
1017                 mpImpl->mpTextEngine->FormatAndUpdate( this );
1018             }
1019 
1020             TextSelection aNewSel = TextPaM( mpImpl->mpTextEngine->mpIMEInfos->aPos.GetPara(), mpImpl->mpTextEngine->mpIMEInfos->aPos.GetIndex()+pData->GetCursorPos() );
1021 			SetSelection( aNewSel );
1022 			SetInsertMode( !pData->IsCursorOverwrite() );
1023 
1024 			if ( pData->IsCursorVisible() )
1025 				ShowCursor();
1026 			else
1027 				HideCursor();
1028 		}
1029 	}
1030 	else if ( rCEvt.GetCommand() == COMMAND_CURSORPOS )
1031 	{
1032         if ( mpImpl->mpTextEngine->mpIMEInfos && mpImpl->mpTextEngine->mpIMEInfos->nLen )
1033 		{
1034 			TextPaM aPaM( GetSelection().GetEnd() );
1035             Rectangle aR1 = mpImpl->mpTextEngine->PaMtoEditCursor( aPaM );
1036 
1037             sal_uInt16 nInputEnd = mpImpl->mpTextEngine->mpIMEInfos->aPos.GetIndex() + mpImpl->mpTextEngine->mpIMEInfos->nLen;
1038 
1039             if ( !mpImpl->mpTextEngine->IsFormatted() )
1040                 mpImpl->mpTextEngine->FormatDoc();
1041 
1042             TEParaPortion* pParaPortion = mpImpl->mpTextEngine->mpTEParaPortions->GetObject( aPaM.GetPara() );
1043 			sal_uInt16 nLine = pParaPortion->GetLineNumber( aPaM.GetIndex(), sal_True );
1044 			TextLine* pLine = pParaPortion->GetLines().GetObject( nLine );
1045 			if ( pLine && ( nInputEnd > pLine->GetEnd() ) )
1046 				nInputEnd = pLine->GetEnd();
1047             Rectangle aR2 = mpImpl->mpTextEngine->PaMtoEditCursor( TextPaM( aPaM.GetPara(), nInputEnd ) );
1048 
1049 			long nWidth = aR2.Left()-aR1.Right();
1050 			aR1.Move( -GetStartDocPos().X(), -GetStartDocPos().Y() );
1051 			GetWindow()->SetCursorRect( &aR1, nWidth );
1052 		}
1053 		else
1054 		{
1055 			GetWindow()->SetCursorRect();
1056 		}
1057 	}
1058 	else
1059 	{
1060         mpImpl->mpSelEngine->Command( rCEvt );
1061 	}
1062 }
1063 
1064 void TextView::ShowCursor( sal_Bool bGotoCursor, sal_Bool bForceVisCursor )
1065 {
1066 	// Die Einstellung hat mehr Gewicht:
1067     if ( !mpImpl->mbAutoScroll )
1068 		bGotoCursor = sal_False;
1069 	ImpShowCursor( bGotoCursor, bForceVisCursor, sal_False );
1070 }
1071 
1072 void TextView::HideCursor()
1073 {
1074     mpImpl->mpCursor->Hide();
1075 }
1076 
1077 void TextView::Scroll( long ndX, long ndY )
1078 {
1079     DBG_ASSERT( mpImpl->mpTextEngine->IsFormatted(), "Scroll: Nicht formatiert!" );
1080 
1081 	if ( !ndX && !ndY )
1082 		return;
1083 
1084     Point aNewStartPos( mpImpl->maStartDocPos );
1085 
1086 	// Vertical:
1087 	aNewStartPos.Y() -= ndY;
1088 	if ( aNewStartPos.Y() < 0 )
1089 		aNewStartPos.Y() = 0;
1090 
1091 	// Horizontal:
1092 	aNewStartPos.X() -= ndX;
1093 	if ( aNewStartPos.X() < 0 )
1094 		aNewStartPos.X() = 0;
1095 
1096     long nDiffX = mpImpl->maStartDocPos.X() - aNewStartPos.X();
1097     long nDiffY = mpImpl->maStartDocPos.Y() - aNewStartPos.Y();
1098 
1099 	if ( nDiffX || nDiffY )
1100 	{
1101         sal_Bool bVisCursor = mpImpl->mpCursor->IsVisible();
1102         mpImpl->mpCursor->Hide();
1103         mpImpl->mpWindow->Update();
1104         mpImpl->maStartDocPos = aNewStartPos;
1105 
1106         if ( mpImpl->mpTextEngine->IsRightToLeft() )
1107             nDiffX = -nDiffX;
1108         mpImpl->mpWindow->Scroll( nDiffX, nDiffY );
1109         mpImpl->mpWindow->Update();
1110         mpImpl->mpCursor->SetPos( mpImpl->mpCursor->GetPos() + Point( nDiffX, nDiffY ) );
1111         if ( bVisCursor && !mpImpl->mbReadOnly )
1112             mpImpl->mpCursor->Show();
1113 	}
1114 
1115     mpImpl->mpTextEngine->Broadcast( TextHint( TEXT_HINT_VIEWSCROLLED ) );
1116 }
1117 
1118 void TextView::Undo()
1119 {
1120     mpImpl->mpTextEngine->SetActiveView( this );
1121     mpImpl->mpTextEngine->GetUndoManager().Undo();
1122 }
1123 
1124 void TextView::Redo()
1125 {
1126     mpImpl->mpTextEngine->SetActiveView( this );
1127     mpImpl->mpTextEngine->GetUndoManager().Redo();
1128 }
1129 
1130 void TextView::Cut()
1131 {
1132     mpImpl->mpTextEngine->UndoActionStart();
1133 	Copy();
1134 	DeleteSelected();
1135     mpImpl->mpTextEngine->UndoActionEnd();
1136 }
1137 
1138 void TextView::Copy( uno::Reference< datatransfer::clipboard::XClipboard >& rxClipboard )
1139 {
1140 	if ( rxClipboard.is() )
1141 	{
1142 		TETextDataObject* pDataObj = new TETextDataObject( GetSelected() );
1143 
1144         if ( mpImpl->mpTextEngine->HasAttrib( TEXTATTR_HYPERLINK ) )  // Dann auch als HTML
1145             mpImpl->mpTextEngine->Write( pDataObj->GetHTMLStream(), &mpImpl->maSelection, sal_True );
1146 
1147 		const sal_uInt32 nRef = Application::ReleaseSolarMutex();
1148 
1149         try
1150 		{
1151 		    rxClipboard->setContents( pDataObj, NULL );
1152 
1153             uno::Reference< datatransfer::clipboard::XFlushableClipboard > xFlushableClipboard( rxClipboard, uno::UNO_QUERY );
1154 		    if( xFlushableClipboard.is() )
1155 			    xFlushableClipboard->flushClipboard();
1156 		}
1157 		catch( const ::com::sun::star::uno::Exception& )
1158 		{
1159 		}
1160 
1161 		Application::AcquireSolarMutex( nRef );
1162 	}
1163 }
1164 
1165 void TextView::Copy()
1166 {
1167     uno::Reference<datatransfer::clipboard::XClipboard> aClipboard(GetWindow()->GetClipboard());
1168     Copy( aClipboard );
1169 }
1170 
1171 void TextView::Paste( uno::Reference< datatransfer::clipboard::XClipboard >& rxClipboard )
1172 {
1173 	if ( rxClipboard.is() )
1174 	{
1175         uno::Reference< datatransfer::XTransferable > xDataObj;
1176 
1177         const sal_uInt32 nRef = Application::ReleaseSolarMutex();
1178 
1179         try
1180 		{
1181 		    xDataObj = rxClipboard->getContents();
1182 		}
1183 		catch( const ::com::sun::star::uno::Exception& )
1184 		{
1185 		}
1186 
1187         Application::AcquireSolarMutex( nRef );
1188 
1189         if ( xDataObj.is() )
1190 		{
1191 			datatransfer::DataFlavor aFlavor;
1192 			SotExchange::GetFormatDataFlavor( SOT_FORMAT_STRING, aFlavor );
1193 			if ( xDataObj->isDataFlavorSupported( aFlavor ) )
1194 			{
1195                 try
1196                 {
1197                     uno::Any aData = xDataObj->getTransferData( aFlavor );
1198                     ::rtl::OUString aText;
1199                     aData >>= aText;
1200                     bool bWasTruncated = false;
1201                     if( mpImpl->mpTextEngine->GetMaxTextLen() != 0 )
1202                         bWasTruncated = ImplTruncateNewText( aText );
1203                     InsertNewText( aText, sal_False );
1204                     mpImpl->mpTextEngine->Broadcast( TextHint( TEXT_HINT_MODIFIED ) );
1205 
1206                     if( bWasTruncated )
1207                         Edit::ShowTruncationWarning( mpImpl->mpWindow );
1208                 }
1209                 catch( const ::com::sun::star::datatransfer::UnsupportedFlavorException& )
1210                 {
1211                 }
1212 			}
1213 		}
1214 	}
1215 }
1216 
1217 void TextView::Paste()
1218 {
1219     uno::Reference<datatransfer::clipboard::XClipboard> aClipboard(GetWindow()->GetClipboard());
1220     Paste( aClipboard );
1221 }
1222 
1223 String TextView::GetSelected()
1224 {
1225     return GetSelected( GetSystemLineEnd() );
1226 }
1227 
1228 String TextView::GetSelected( LineEnd aSeparator )
1229 {
1230     return mpImpl->mpTextEngine->GetText( mpImpl->maSelection, aSeparator );
1231 }
1232 
1233 void TextView::SetInsertMode( sal_Bool bInsert )
1234 {
1235     if ( mpImpl->mbInsertMode != bInsert )
1236 	{
1237         mpImpl->mbInsertMode = bInsert;
1238         ShowCursor( mpImpl->mbAutoScroll, sal_False );
1239 	}
1240 }
1241 
1242 void TextView::SetReadOnly( sal_Bool bReadOnly )
1243 {
1244     if ( mpImpl->mbReadOnly != bReadOnly )
1245 	{
1246         mpImpl->mbReadOnly = bReadOnly;
1247         if ( !mpImpl->mbReadOnly )
1248             ShowCursor( mpImpl->mbAutoScroll, sal_False );
1249 		else
1250 			HideCursor();
1251 
1252         GetWindow()->SetInputContext( InputContext( mpImpl->mpTextEngine->GetFont(), bReadOnly ? INPUTCONTEXT_TEXT|INPUTCONTEXT_EXTTEXTINPUT : 0 ) );
1253 	}
1254 }
1255 
1256 TextSelection TextView::ImpMoveCursor( const KeyEvent& rKeyEvent )
1257 {
1258 	// Eigentlich nur bei Up/Down noetig, aber was solls.
1259     mpImpl->mpTextEngine->CheckIdleFormatter();
1260 
1261     TextPaM aPaM( mpImpl->maSelection.GetEnd() );
1262 	TextPaM aOldEnd( aPaM );
1263 
1264     TextDirectionality eTextDirection = TextDirectionality_LeftToRight_TopToBottom;
1265     if ( mpImpl->mpTextEngine->IsRightToLeft() )
1266         eTextDirection = TextDirectionality_RightToLeft_TopToBottom;
1267 
1268     KeyEvent aTranslatedKeyEvent = rKeyEvent.LogicalTextDirectionality( eTextDirection );
1269 
1270     sal_Bool bCtrl = aTranslatedKeyEvent.GetKeyCode().IsMod1() ? sal_True : sal_False;
1271 	sal_uInt16 nCode = aTranslatedKeyEvent.GetKeyCode().GetCode();
1272 
1273     bool bSelect = aTranslatedKeyEvent.GetKeyCode().IsShift();
1274 	switch ( nCode )
1275 	{
1276 		case KEY_UP:		aPaM = CursorUp( aPaM );
1277 							break;
1278 		case KEY_DOWN:		aPaM = CursorDown( aPaM );
1279 							break;
1280 		case KEY_HOME:		aPaM = bCtrl ? CursorStartOfDoc() : CursorStartOfLine( aPaM );
1281 							break;
1282 		case KEY_END:		aPaM = bCtrl ? CursorEndOfDoc() : CursorEndOfLine( aPaM );
1283 							break;
1284 		case KEY_PAGEUP:	aPaM = bCtrl ? CursorStartOfDoc() : PageUp( aPaM );
1285 							break;
1286 		case KEY_PAGEDOWN:	aPaM = bCtrl ? CursorEndOfDoc() : PageDown( aPaM );
1287 							break;
1288 		case KEY_LEFT:		aPaM = bCtrl ? CursorWordLeft( aPaM ) : CursorLeft( aPaM, aTranslatedKeyEvent.GetKeyCode().IsMod2() ? (sal_uInt16)i18n::CharacterIteratorMode::SKIPCHARACTER : (sal_uInt16)i18n::CharacterIteratorMode::SKIPCELL );
1289 							break;
1290 		case KEY_RIGHT: 	aPaM = bCtrl ? CursorWordRight( aPaM ) : CursorRight( aPaM, aTranslatedKeyEvent.GetKeyCode().IsMod2() ? (sal_uInt16)i18n::CharacterIteratorMode::SKIPCHARACTER : (sal_uInt16)i18n::CharacterIteratorMode::SKIPCELL );
1291 							break;
1292         case com::sun::star::awt::Key::SELECT_WORD_FORWARD:
1293                             bSelect = true; // fallthrough intentional
1294         case com::sun::star::awt::Key::MOVE_WORD_FORWARD:
1295                             aPaM = CursorWordRight( aPaM );
1296                             break;
1297         case com::sun::star::awt::Key::SELECT_WORD_BACKWARD:
1298                             bSelect = true; // fallthrough intentional
1299         case com::sun::star::awt::Key::MOVE_WORD_BACKWARD:
1300                             aPaM = CursorWordLeft( aPaM );
1301                             break;
1302         case com::sun::star::awt::Key::SELECT_TO_BEGIN_OF_LINE:
1303                             bSelect = true; // fallthrough intentional
1304         case com::sun::star::awt::Key::MOVE_TO_BEGIN_OF_LINE:
1305                             aPaM = CursorStartOfLine( aPaM );
1306                             break;
1307         case com::sun::star::awt::Key::SELECT_TO_END_OF_LINE:
1308                             bSelect = true; // fallthrough intentional
1309         case com::sun::star::awt::Key::MOVE_TO_END_OF_LINE:
1310                             aPaM = CursorEndOfLine( aPaM );
1311                             break;
1312         case com::sun::star::awt::Key::SELECT_TO_BEGIN_OF_PARAGRAPH:
1313                             bSelect = true; // falltthrough intentional
1314         case com::sun::star::awt::Key::MOVE_TO_BEGIN_OF_PARAGRAPH:
1315                             aPaM = CursorStartOfParagraph( aPaM );
1316                             break;
1317         case com::sun::star::awt::Key::SELECT_TO_END_OF_PARAGRAPH:
1318                             bSelect = true; // falltthrough intentional
1319         case com::sun::star::awt::Key::MOVE_TO_END_OF_PARAGRAPH:
1320                             aPaM = CursorEndOfParagraph( aPaM );
1321                             break;
1322         case com::sun::star::awt::Key::SELECT_TO_BEGIN_OF_DOCUMENT:
1323                             bSelect = true; // falltthrough intentional
1324         case com::sun::star::awt::Key::MOVE_TO_BEGIN_OF_DOCUMENT:
1325                             aPaM = CursorStartOfDoc();
1326                             break;
1327         case com::sun::star::awt::Key::SELECT_TO_END_OF_DOCUMENT:
1328                             bSelect = true; // falltthrough intentional
1329         case com::sun::star::awt::Key::MOVE_TO_END_OF_DOCUMENT:
1330                             aPaM = CursorEndOfDoc();
1331                             break;
1332 	}
1333 
1334 	// Bewirkt evtl. ein CreateAnchor oder Deselection all
1335     mpImpl->mpSelEngine->CursorPosChanging( bSelect, aTranslatedKeyEvent.GetKeyCode().IsMod1() );
1336 
1337 	if ( aOldEnd != aPaM )
1338 	{
1339         mpImpl->mpTextEngine->CursorMoved( aOldEnd.GetPara() );
1340 
1341 
1342         TextSelection aOldSelection( mpImpl->maSelection );
1343         TextSelection aNewSelection( mpImpl->maSelection );
1344 		aNewSelection.GetEnd() = aPaM;
1345 		if ( bSelect )
1346 		{
1347 			// Dann wird die Selektion erweitert...
1348             ImpSetSelection( aNewSelection );
1349 			ShowSelection( TextSelection( aOldEnd, aPaM ) );
1350 		}
1351 		else
1352 		{
1353 			aNewSelection.GetStart() = aPaM;
1354             ImpSetSelection( aNewSelection );
1355 		}
1356 	}
1357 
1358     return mpImpl->maSelection;
1359 }
1360 
1361 void TextView::InsertText( const XubString& rStr, sal_Bool bSelect )
1362 {
1363     InsertNewText( rStr, bSelect );
1364 }
1365 
1366 void TextView::InsertNewText( const rtl::OUString& rStr, sal_Bool bSelect )
1367 {
1368 //	HideSelection();
1369     mpImpl->mpTextEngine->UndoActionStart();
1370 
1371     /* #i87633#
1372     break inserted text into chunks that fit into the underlying String
1373     based API (which has a maximum length of 65534 elements
1374 
1375     note: this will of course still cause problems for lines longer than those
1376     65534 elements, but those cases will hopefully be few.
1377     In the long run someone should switch the TextEngine to OUString instead of String
1378     */
1379     sal_Int32 nLen = rStr.getLength();
1380     sal_Int32 nPos = 0;
1381     while( nLen )
1382     {
1383         sal_Int32 nChunkLen = nLen > 65534 ? 65534 : nLen;
1384         String aChunk( rStr.copy( nPos, nChunkLen ) );
1385 
1386         TextSelection aNewSel( mpImpl->maSelection );
1387 
1388         TextPaM aPaM = mpImpl->mpTextEngine->ImpInsertText( mpImpl->maSelection, aChunk );
1389 
1390         if ( bSelect )
1391         {
1392             aNewSel.Justify();
1393             aNewSel.GetEnd() = aPaM;
1394         }
1395         else
1396         {
1397             aNewSel = aPaM;
1398         }
1399 
1400         ImpSetSelection( aNewSel );
1401         nLen -= nChunkLen;
1402         nPos += nChunkLen;
1403     }
1404     mpImpl->mpTextEngine->UndoActionEnd();
1405 
1406     mpImpl->mpTextEngine->FormatAndUpdate( this );
1407 }
1408 
1409 /*
1410 void TextView::InsertText( const XubString& rStr, sal_Bool bSelect )
1411 {
1412 //	HideSelection();
1413 
1414     TextSelection aNewSel( mpImpl->maSelection );
1415 
1416     mpImpl->mpTextEngine->UndoActionStart();
1417     TextPaM aPaM = mpImpl->mpTextEngine->ImpInsertText( mpImpl->maSelection, rStr );
1418     mpImpl->mpTextEngine->UndoActionEnd();
1419 
1420 	if ( bSelect )
1421 	{
1422 		aNewSel.Justify();
1423 		aNewSel.GetEnd() = aPaM;
1424 	}
1425 	else
1426     {
1427 		aNewSel = aPaM;
1428     }
1429 
1430     ImpSetSelection( aNewSel );
1431 
1432     mpImpl->mpTextEngine->FormatAndUpdate( this );
1433 }
1434 */
1435 
1436 // OLD
1437 TextPaM TextView::CursorLeft( const TextPaM& rPaM, sal_Bool bWordMode )
1438 {
1439     return bWordMode ? CursorWordLeft( rPaM ) : CursorLeft( rPaM, (sal_uInt16)i18n::CharacterIteratorMode::SKIPCELL );
1440 
1441     // Remove (sal_uInt16) typecasts in this file when removing this method!
1442 }
1443 
1444 // OLD
1445 TextPaM TextView::CursorRight( const TextPaM& rPaM, sal_Bool bWordMode )
1446 {
1447     return bWordMode ? CursorWordRight( rPaM ) : CursorRight( rPaM, (sal_uInt16)i18n::CharacterIteratorMode::SKIPCELL );
1448 
1449     // Remove (sal_uInt16) typecasts in this file when removing this method!
1450 }
1451 
1452 TextPaM TextView::CursorLeft( const TextPaM& rPaM, sal_uInt16 nCharacterIteratorMode )
1453 {
1454 	TextPaM aPaM( rPaM );
1455 
1456 	if ( aPaM.GetIndex() )
1457 	{
1458         TextNode* pNode = mpImpl->mpTextEngine->mpDoc->GetNodes().GetObject( aPaM.GetPara() );
1459         uno::Reference < i18n::XBreakIterator > xBI = mpImpl->mpTextEngine->GetBreakIterator();
1460 		sal_Int32 nCount = 1;
1461         aPaM.GetIndex() = (sal_uInt16)xBI->previousCharacters( pNode->GetText(), aPaM.GetIndex(), mpImpl->mpTextEngine->GetLocale(), nCharacterIteratorMode, nCount, nCount );
1462 	}
1463 	else if ( aPaM.GetPara() )
1464 	{
1465 		aPaM.GetPara()--;
1466         TextNode* pNode = mpImpl->mpTextEngine->mpDoc->GetNodes().GetObject( aPaM.GetPara() );
1467 		aPaM.GetIndex() = pNode->GetText().Len();
1468 	}
1469 	return aPaM;
1470 }
1471 
1472 TextPaM TextView::CursorRight( const TextPaM& rPaM, sal_uInt16 nCharacterIteratorMode )
1473 {
1474 	TextPaM aPaM( rPaM );
1475 
1476     TextNode* pNode = mpImpl->mpTextEngine->mpDoc->GetNodes().GetObject( aPaM.GetPara() );
1477 	if ( aPaM.GetIndex() < pNode->GetText().Len() )
1478 	{
1479         uno::Reference < i18n::XBreakIterator > xBI = mpImpl->mpTextEngine->GetBreakIterator();
1480 		sal_Int32 nCount = 1;
1481         aPaM.GetIndex() = (sal_uInt16)xBI->nextCharacters( pNode->GetText(), aPaM.GetIndex(), mpImpl->mpTextEngine->GetLocale(), nCharacterIteratorMode, nCount, nCount );
1482 	}
1483     else if ( aPaM.GetPara() < ( mpImpl->mpTextEngine->mpDoc->GetNodes().Count()-1) )
1484 	{
1485 		aPaM.GetPara()++;
1486 		aPaM.GetIndex() = 0;
1487 	}
1488 
1489 	return aPaM;
1490 }
1491 
1492 
1493 TextPaM TextView::CursorWordLeft( const TextPaM& rPaM )
1494 {
1495 	TextPaM aPaM( rPaM );
1496 
1497 	if ( aPaM.GetIndex() )
1498 	{
1499         TextNode* pNode = mpImpl->mpTextEngine->mpDoc->GetNodes().GetObject( aPaM.GetPara() );
1500         uno::Reference < i18n::XBreakIterator > xBI = mpImpl->mpTextEngine->GetBreakIterator();
1501         i18n::Boundary aBoundary = xBI->getWordBoundary( pNode->GetText(), rPaM.GetIndex(), mpImpl->mpTextEngine->GetLocale(), i18n::WordType::ANYWORD_IGNOREWHITESPACES, sal_True );
1502 		if ( aBoundary.startPos >= rPaM.GetIndex() )
1503             aBoundary = xBI->previousWord( pNode->GetText(), rPaM.GetIndex(), mpImpl->mpTextEngine->GetLocale(), i18n::WordType::ANYWORD_IGNOREWHITESPACES );
1504 		aPaM.GetIndex() = ( aBoundary.startPos != -1 ) ? (sal_uInt16)aBoundary.startPos : 0;
1505 	}
1506 	else if ( aPaM.GetPara() )
1507 	{
1508 		aPaM.GetPara()--;
1509         TextNode* pNode = mpImpl->mpTextEngine->mpDoc->GetNodes().GetObject( aPaM.GetPara() );
1510 		aPaM.GetIndex() = pNode->GetText().Len();
1511 	}
1512 	return aPaM;
1513 }
1514 
1515 
1516 TextPaM TextView::CursorWordRight( const TextPaM& rPaM )
1517 {
1518 	TextPaM aPaM( rPaM );
1519 
1520     TextNode* pNode = mpImpl->mpTextEngine->mpDoc->GetNodes().GetObject( aPaM.GetPara() );
1521 	if ( aPaM.GetIndex() < pNode->GetText().Len() )
1522 	{
1523         uno::Reference < i18n::XBreakIterator > xBI = mpImpl->mpTextEngine->GetBreakIterator();
1524         i18n::Boundary aBoundary = xBI->nextWord(  pNode->GetText(), aPaM.GetIndex(), mpImpl->mpTextEngine->GetLocale(), i18n::WordType::ANYWORD_IGNOREWHITESPACES );
1525 		aPaM.GetIndex() = (sal_uInt16)aBoundary.startPos;
1526 	}
1527     else if ( aPaM.GetPara() < ( mpImpl->mpTextEngine->mpDoc->GetNodes().Count()-1) )
1528 	{
1529 		aPaM.GetPara()++;
1530 		aPaM.GetIndex() = 0;
1531 	}
1532 
1533 	return aPaM;
1534 }
1535 
1536 TextPaM TextView::ImpDelete( sal_uInt8 nMode, sal_uInt8 nDelMode )
1537 {
1538     if ( mpImpl->maSelection.HasRange() )  // dann nur Sel. loeschen
1539         return mpImpl->mpTextEngine->ImpDeleteText( mpImpl->maSelection );
1540 
1541     TextPaM aStartPaM = mpImpl->maSelection.GetStart();
1542 	TextPaM aEndPaM = aStartPaM;
1543 	if ( nMode == DEL_LEFT )
1544 	{
1545 		if ( nDelMode == DELMODE_SIMPLE )
1546 		{
1547 			aEndPaM = CursorLeft( aEndPaM, (sal_uInt16)i18n::CharacterIteratorMode::SKIPCHARACTER );
1548 		}
1549 		else if ( nDelMode == DELMODE_RESTOFWORD )
1550 		{
1551             TextNode* pNode = mpImpl->mpTextEngine->mpDoc->GetNodes().GetObject(  aEndPaM.GetPara() );
1552             uno::Reference < i18n::XBreakIterator > xBI = mpImpl->mpTextEngine->GetBreakIterator();
1553             i18n::Boundary aBoundary = xBI->getWordBoundary( pNode->GetText(), mpImpl->maSelection.GetEnd().GetIndex(), mpImpl->mpTextEngine->GetLocale(), i18n::WordType::ANYWORD_IGNOREWHITESPACES, sal_True );
1554             if ( aBoundary.startPos == mpImpl->maSelection.GetEnd().GetIndex() )
1555                 aBoundary = xBI->previousWord( pNode->GetText(), mpImpl->maSelection.GetEnd().GetIndex(), mpImpl->mpTextEngine->GetLocale(), i18n::WordType::ANYWORD_IGNOREWHITESPACES );
1556 			// #i63506# startPos is -1 when the paragraph starts with a tab
1557 			aEndPaM.GetIndex() = (aBoundary.startPos >= 0) ? (sal_uInt16)aBoundary.startPos : 0;
1558 		}
1559 		else	// DELMODE_RESTOFCONTENT
1560 		{
1561 			if ( aEndPaM.GetIndex() != 0 )
1562 				aEndPaM.GetIndex() = 0;
1563 			else if ( aEndPaM.GetPara() )
1564 			{
1565 				// Absatz davor
1566 				aEndPaM.GetPara()--;
1567 				aEndPaM.GetIndex() = 0;
1568 			}
1569 		}
1570 	}
1571 	else
1572 	{
1573 		if ( nDelMode == DELMODE_SIMPLE )
1574 		{
1575 			aEndPaM = CursorRight( aEndPaM, (sal_uInt16)i18n::CharacterIteratorMode::SKIPCELL );
1576 		}
1577 		else if ( nDelMode == DELMODE_RESTOFWORD )
1578 		{
1579             TextNode* pNode = mpImpl->mpTextEngine->mpDoc->GetNodes().GetObject(  aEndPaM.GetPara() );
1580             uno::Reference < i18n::XBreakIterator > xBI = mpImpl->mpTextEngine->GetBreakIterator();
1581             i18n::Boundary aBoundary = xBI->nextWord( pNode->GetText(), mpImpl->maSelection.GetEnd().GetIndex(), mpImpl->mpTextEngine->GetLocale(), i18n::WordType::ANYWORD_IGNOREWHITESPACES );
1582 			aEndPaM.GetIndex() = (sal_uInt16)aBoundary.startPos;
1583 		}
1584 		else	// DELMODE_RESTOFCONTENT
1585 		{
1586             TextNode* pNode = mpImpl->mpTextEngine->mpDoc->GetNodes().GetObject( aEndPaM.GetPara() );
1587 			if ( aEndPaM.GetIndex() < pNode->GetText().Len() )
1588 				aEndPaM.GetIndex() = pNode->GetText().Len();
1589             else if ( aEndPaM.GetPara() < ( mpImpl->mpTextEngine->mpDoc->GetNodes().Count() - 1 ) )
1590 			{
1591 				// Absatz danach
1592 				aEndPaM.GetPara()++;
1593                 TextNode* pNextNode = mpImpl->mpTextEngine->mpDoc->GetNodes().GetObject( aEndPaM.GetPara() );
1594 				aEndPaM.GetIndex() = pNextNode->GetText().Len();
1595 			}
1596 		}
1597 	}
1598 
1599     return mpImpl->mpTextEngine->ImpDeleteText( TextSelection( aStartPaM, aEndPaM ) );
1600 }
1601 
1602 
1603 
1604 TextPaM TextView::CursorUp( const TextPaM& rPaM )
1605 {
1606 	TextPaM aPaM( rPaM );
1607 
1608 	long nX;
1609     if ( mpImpl->mnTravelXPos == TRAVEL_X_DONTKNOW )
1610 	{
1611         nX = mpImpl->mpTextEngine->GetEditCursor( rPaM, sal_False ).Left();
1612         mpImpl->mnTravelXPos = (sal_uInt16)nX+1;
1613 	}
1614 	else
1615         nX = mpImpl->mnTravelXPos;
1616 
1617     TEParaPortion* pPPortion = mpImpl->mpTextEngine->mpTEParaPortions->GetObject( rPaM.GetPara() );
1618 	sal_uInt16 nLine = pPPortion->GetLineNumber( rPaM.GetIndex(), sal_False );
1619 	if ( nLine )	// gleicher Absatz
1620 	{
1621         sal_uInt16 nCharPos = mpImpl->mpTextEngine->GetCharPos( rPaM.GetPara(), nLine-1, nX );
1622 		aPaM.GetIndex() = nCharPos;
1623 		// Wenn davor eine autom.Umgebrochene Zeile, und ich muss genau an das
1624 		// Ende dieser Zeile, landet der Cursor in der aktuellen Zeile am Anfang
1625 		// Siehe Problem: Letztes Zeichen einer autom.umgebr. Zeile = Cursor
1626 		TextLine* pLine = pPPortion->GetLines().GetObject( nLine - 1 );
1627 		if ( aPaM.GetIndex() && ( aPaM.GetIndex() == pLine->GetEnd() ) )
1628 			aPaM.GetIndex()--;
1629 	}
1630 	else if ( rPaM.GetPara() )	// vorheriger Absatz
1631 	{
1632 		aPaM.GetPara()--;
1633         pPPortion = mpImpl->mpTextEngine->mpTEParaPortions->GetObject( aPaM.GetPara() );
1634 		sal_uInt16 nL = pPPortion->GetLines().Count() - 1;
1635         sal_uInt16 nCharPos = mpImpl->mpTextEngine->GetCharPos( aPaM.GetPara(), nL, nX+1 );
1636 		aPaM.GetIndex() = nCharPos;
1637 	}
1638 
1639 	return aPaM;
1640 }
1641 
1642 TextPaM TextView::CursorDown( const TextPaM& rPaM )
1643 {
1644 	TextPaM aPaM( rPaM );
1645 
1646 	long nX;
1647     if ( mpImpl->mnTravelXPos == TRAVEL_X_DONTKNOW )
1648 	{
1649         nX = mpImpl->mpTextEngine->GetEditCursor( rPaM, sal_False ).Left();
1650         mpImpl->mnTravelXPos = (sal_uInt16)nX+1;
1651 	}
1652 	else
1653         nX = mpImpl->mnTravelXPos;
1654 
1655     TEParaPortion* pPPortion = mpImpl->mpTextEngine->mpTEParaPortions->GetObject( rPaM.GetPara() );
1656 	sal_uInt16 nLine = pPPortion->GetLineNumber( rPaM.GetIndex(), sal_False );
1657 	if ( nLine < ( pPPortion->GetLines().Count() - 1 ) )
1658 	{
1659         sal_uInt16 nCharPos = mpImpl->mpTextEngine->GetCharPos( rPaM.GetPara(), nLine+1, nX );
1660 		aPaM.GetIndex() = nCharPos;
1661 
1662 		// Sonderbehandlung siehe CursorUp...
1663 		TextLine* pLine = pPPortion->GetLines().GetObject( nLine + 1 );
1664 		if ( ( aPaM.GetIndex() == pLine->GetEnd() ) && ( aPaM.GetIndex() > pLine->GetStart() ) && aPaM.GetIndex() < pPPortion->GetNode()->GetText().Len() )
1665 			aPaM.GetIndex()--;
1666 	}
1667     else if ( rPaM.GetPara() < ( mpImpl->mpTextEngine->mpDoc->GetNodes().Count() - 1 ) )   // naechster Absatz
1668 	{
1669 		aPaM.GetPara()++;
1670         pPPortion = mpImpl->mpTextEngine->mpTEParaPortions->GetObject( aPaM.GetPara() );
1671         sal_uInt16 nCharPos = mpImpl->mpTextEngine->GetCharPos( aPaM.GetPara(), 0, nX+1 );
1672 		aPaM.GetIndex() = nCharPos;
1673 		TextLine* pLine = pPPortion->GetLines().GetObject( 0 );
1674 		if ( ( aPaM.GetIndex() == pLine->GetEnd() ) && ( aPaM.GetIndex() > pLine->GetStart() ) && ( pPPortion->GetLines().Count() > 1 ) )
1675 			aPaM.GetIndex()--;
1676 	}
1677 
1678 	return aPaM;
1679 }
1680 
1681 TextPaM TextView::CursorStartOfLine( const TextPaM& rPaM )
1682 {
1683 	TextPaM aPaM( rPaM );
1684 
1685     TEParaPortion* pPPortion = mpImpl->mpTextEngine->mpTEParaPortions->GetObject( rPaM.GetPara() );
1686 	sal_uInt16 nLine = pPPortion->GetLineNumber( aPaM.GetIndex(), sal_False );
1687 	TextLine* pLine = pPPortion->GetLines().GetObject( nLine );
1688 	aPaM.GetIndex() = pLine->GetStart();
1689 
1690 	return aPaM;
1691 }
1692 
1693 TextPaM TextView::CursorEndOfLine( const TextPaM& rPaM )
1694 {
1695 	TextPaM aPaM( rPaM );
1696 
1697     TEParaPortion* pPPortion = mpImpl->mpTextEngine->mpTEParaPortions->GetObject( rPaM.GetPara() );
1698 	sal_uInt16 nLine = pPPortion->GetLineNumber( aPaM.GetIndex(), sal_False );
1699 	TextLine* pLine = pPPortion->GetLines().GetObject( nLine );
1700 	aPaM.GetIndex() = pLine->GetEnd();
1701 
1702 	if ( pLine->GetEnd() > pLine->GetStart() )	// Leerzeile
1703 	{
1704 		xub_Unicode cLastChar = pPPortion->GetNode()->GetText().GetChar((sal_uInt16)(aPaM.GetIndex()-1) );
1705 		if ( ( cLastChar == ' ' ) && ( aPaM.GetIndex() != pPPortion->GetNode()->GetText().Len() ) )
1706 		{
1707 			// Bei einem Blank in einer autom. umgebrochenen Zeile macht es Sinn,
1708 			// davor zu stehen, da der Anwender hinter das Wort will.
1709 			// Wenn diese geaendert wird, Sonderbehandlung fuer Pos1 nach End!
1710 			aPaM.GetIndex()--;
1711 		}
1712 	}
1713 	return aPaM;
1714 }
1715 
1716 TextPaM TextView::CursorStartOfParagraph( const TextPaM& rPaM )
1717 {
1718 	TextPaM aPaM( rPaM );
1719 	aPaM.GetIndex() = 0;
1720 	return aPaM;
1721 }
1722 
1723 TextPaM TextView::CursorEndOfParagraph( const TextPaM& rPaM )
1724 {
1725     TextNode* pNode = mpImpl->mpTextEngine->mpDoc->GetNodes().GetObject( rPaM.GetPara() );
1726 	TextPaM aPaM( rPaM );
1727 	aPaM.GetIndex() = pNode->GetText().Len();
1728 	return aPaM;
1729 }
1730 
1731 TextPaM TextView::CursorStartOfDoc()
1732 {
1733 	TextPaM aPaM( 0, 0 );
1734 	return aPaM;
1735 }
1736 
1737 TextPaM TextView::CursorEndOfDoc()
1738 {
1739     sal_uLong nNode = mpImpl->mpTextEngine->mpDoc->GetNodes().Count() - 1;
1740     TextNode* pNode = mpImpl->mpTextEngine->mpDoc->GetNodes().GetObject( nNode );
1741 	TextPaM aPaM( nNode, pNode->GetText().Len() );
1742 	return aPaM;
1743 }
1744 
1745 TextPaM TextView::PageUp( const TextPaM& rPaM )
1746 {
1747     Rectangle aRec = mpImpl->mpTextEngine->PaMtoEditCursor( rPaM );
1748 	Point aTopLeft = aRec.TopLeft();
1749     aTopLeft.Y() -= mpImpl->mpWindow->GetOutputSizePixel().Height() * 9/10;
1750 	aTopLeft.X() += 1;
1751 	if ( aTopLeft.Y() < 0 )
1752 		aTopLeft.Y() = 0;
1753 
1754     TextPaM aPaM = mpImpl->mpTextEngine->GetPaM( aTopLeft );
1755 	return aPaM;
1756 }
1757 
1758 TextPaM TextView::PageDown( const TextPaM& rPaM )
1759 {
1760     Rectangle aRec = mpImpl->mpTextEngine->PaMtoEditCursor( rPaM );
1761 	Point aBottomRight = aRec.BottomRight();
1762     aBottomRight.Y() += mpImpl->mpWindow->GetOutputSizePixel().Height() * 9/10;
1763 	aBottomRight.X() += 1;
1764     long nHeight = mpImpl->mpTextEngine->GetTextHeight();
1765 	if ( aBottomRight.Y() > nHeight )
1766 		aBottomRight.Y() = nHeight-1;
1767 
1768     TextPaM aPaM = mpImpl->mpTextEngine->GetPaM( aBottomRight );
1769 	return aPaM;
1770 }
1771 
1772 void TextView::ImpShowCursor( sal_Bool bGotoCursor, sal_Bool bForceVisCursor, sal_Bool bSpecial )
1773 {
1774     if ( mpImpl->mpTextEngine->IsFormatting() )
1775 		return;
1776     if ( mpImpl->mpTextEngine->GetUpdateMode() == sal_False )
1777 		return;
1778     if ( mpImpl->mpTextEngine->IsInUndo() )
1779 		return;
1780 
1781     mpImpl->mpTextEngine->CheckIdleFormatter();
1782     if ( !mpImpl->mpTextEngine->IsFormatted() )
1783         mpImpl->mpTextEngine->FormatAndUpdate( this );
1784 
1785 
1786     TextPaM aPaM( mpImpl->maSelection.GetEnd() );
1787     Rectangle aEditCursor = mpImpl->mpTextEngine->PaMtoEditCursor( aPaM, bSpecial );
1788 
1789     // Remember that we placed the cursor behind the last character of a line
1790     mpImpl->mbCursorAtEndOfLine = false;
1791     if( bSpecial )
1792     {
1793         TEParaPortion* pParaPortion = mpImpl->mpTextEngine->mpTEParaPortions->GetObject( aPaM.GetPara() );
1794         mpImpl->mbCursorAtEndOfLine =
1795             pParaPortion->GetLineNumber( aPaM.GetIndex(), sal_True ) != pParaPortion->GetLineNumber( aPaM.GetIndex(), sal_False );
1796     }
1797 
1798     if ( !IsInsertMode() && !mpImpl->maSelection.HasRange() )
1799 	{
1800         TextNode* pNode = mpImpl->mpTextEngine->mpDoc->GetNodes().GetObject( aPaM.GetPara() );
1801 		if ( pNode->GetText().Len() && ( aPaM.GetIndex() < pNode->GetText().Len() ) )
1802 		{
1803             // If we are behind a portion, and the next portion has other direction, we must change position...
1804             aEditCursor.Left() = aEditCursor.Right() = mpImpl->mpTextEngine->GetEditCursor( aPaM, sal_False, sal_True ).Left();
1805 
1806             TEParaPortion* pParaPortion = mpImpl->mpTextEngine->mpTEParaPortions->GetObject( aPaM.GetPara() );
1807 
1808             sal_uInt16 nTextPortionStart = 0;
1809             sal_uInt16 nTextPortion = pParaPortion->GetTextPortions().FindPortion( aPaM.GetIndex(), nTextPortionStart, sal_True );
1810             TETextPortion* pTextPortion = pParaPortion->GetTextPortions().GetObject( nTextPortion );
1811             if ( pTextPortion->GetKind() == PORTIONKIND_TAB )
1812             {
1813                 if ( mpImpl->mpTextEngine->IsRightToLeft() )
1814                 {
1815 
1816                 }
1817 		        aEditCursor.Right() += pTextPortion->GetWidth();
1818             }
1819             else
1820             {
1821                 TextPaM aNext = CursorRight( TextPaM( aPaM.GetPara(), aPaM.GetIndex() ), (sal_uInt16)i18n::CharacterIteratorMode::SKIPCELL );
1822                 aEditCursor.Right() = mpImpl->mpTextEngine->GetEditCursor( aNext, sal_True ).Left();
1823             }
1824 		}
1825 	}
1826 
1827     Size aOutSz = mpImpl->mpWindow->GetOutputSizePixel();
1828 	if ( aEditCursor.GetHeight() > aOutSz.Height() )
1829 		aEditCursor.Bottom() = aEditCursor.Top() + aOutSz.Height() - 1;
1830 
1831     aEditCursor.Left() -= 1;
1832 
1833 	if ( bGotoCursor
1834         // #i81283# protext maStartDocPos against initialization problems
1835         && aOutSz.Width() && aOutSz.Height()
1836     )
1837 	{
1838         long nVisStartY = mpImpl->maStartDocPos.Y();
1839         long nVisEndY = mpImpl->maStartDocPos.Y() + aOutSz.Height();
1840         long nVisStartX = mpImpl->maStartDocPos.X();
1841         long nVisEndX = mpImpl->maStartDocPos.X() + aOutSz.Width();
1842 		long nMoreX = aOutSz.Width() / 4;
1843 
1844         Point aNewStartPos( mpImpl->maStartDocPos );
1845 
1846 		if ( aEditCursor.Bottom() > nVisEndY )
1847 		{
1848 			aNewStartPos.Y() += ( aEditCursor.Bottom() - nVisEndY );
1849 		}
1850 		else if ( aEditCursor.Top() < nVisStartY )
1851 		{
1852 			aNewStartPos.Y() -= ( nVisStartY - aEditCursor.Top() );
1853 		}
1854 
1855 		if ( aEditCursor.Right() >= nVisEndX )
1856 		{
1857 			aNewStartPos.X() += ( aEditCursor.Right() - nVisEndX );
1858 
1859 			// Darfs ein bischen mehr sein?
1860 			aNewStartPos.X() += nMoreX;
1861 		}
1862 		else if ( aEditCursor.Left() <= nVisStartX )
1863 		{
1864 			aNewStartPos.X() -= ( nVisStartX - aEditCursor.Left() );
1865 
1866 			// Darfs ein bischen mehr sein?
1867 			aNewStartPos.X() -= nMoreX;
1868 		}
1869 
1870 		// X kann durch das 'bischen mehr' falsch sein:
1871 //      sal_uLong nMaxTextWidth = mpImpl->mpTextEngine->GetMaxTextWidth();
1872 //		if ( !nMaxTextWidth || ( nMaxTextWidth > 0x7FFFFFFF ) )
1873 //			nMaxTextWidth = 0x7FFFFFFF;
1874 //		long nMaxX = (long)nMaxTextWidth - aOutSz.Width();
1875         long nMaxX = mpImpl->mpTextEngine->CalcTextWidth() - aOutSz.Width();
1876 		if ( nMaxX < 0 )
1877 			nMaxX = 0;
1878 
1879 		if ( aNewStartPos.X() < 0 )
1880 			aNewStartPos.X() = 0;
1881 		else if ( aNewStartPos.X() > nMaxX )
1882 			aNewStartPos.X() = nMaxX;
1883 
1884 		// Y sollte nicht weiter unten als noetig liegen:
1885         long nYMax = mpImpl->mpTextEngine->GetTextHeight() - aOutSz.Height();
1886 		if ( nYMax < 0 )
1887 			nYMax = 0;
1888 		if ( aNewStartPos.Y() > nYMax )
1889 			aNewStartPos.Y() = nYMax;
1890 
1891         if ( aNewStartPos != mpImpl->maStartDocPos )
1892             Scroll( -(aNewStartPos.X() - mpImpl->maStartDocPos.X()), -(aNewStartPos.Y() - mpImpl->maStartDocPos.Y()) );
1893 	}
1894 
1895     if ( aEditCursor.Right() < aEditCursor.Left() )
1896     {
1897         long n = aEditCursor.Left();
1898         aEditCursor.Left() = aEditCursor.Right();
1899         aEditCursor.Right() = n;
1900     }
1901 
1902     Point aPoint( GetWindowPos( !mpImpl->mpTextEngine->IsRightToLeft() ? aEditCursor.TopLeft() : aEditCursor.TopRight() ) );
1903     mpImpl->mpCursor->SetPos( aPoint );
1904     mpImpl->mpCursor->SetSize( aEditCursor.GetSize() );
1905     if ( bForceVisCursor && mpImpl->mbCursorEnabled )
1906         mpImpl->mpCursor->Show();
1907 }
1908 
1909 sal_Bool TextView::SetCursorAtPoint( const Point& rPosPixel )
1910 {
1911     mpImpl->mpTextEngine->CheckIdleFormatter();
1912 
1913 	Point aDocPos = GetDocPos( rPosPixel );
1914 
1915     TextPaM aPaM = mpImpl->mpTextEngine->GetPaM( aDocPos );
1916 
1917 	// aTmpNewSel: Diff zwischen alt und neu, nicht die neue Selektion
1918     TextSelection aTmpNewSel( mpImpl->maSelection.GetEnd(), aPaM );
1919     TextSelection aNewSel( mpImpl->maSelection );
1920 	aNewSel.GetEnd() = aPaM;
1921 
1922     if ( !mpImpl->mpSelEngine->HasAnchor() )
1923 	{
1924         if ( mpImpl->maSelection.GetStart() != aPaM )
1925             mpImpl->mpTextEngine->CursorMoved( mpImpl->maSelection.GetStart().GetPara() );
1926 		aNewSel.GetStart() = aPaM;
1927         ImpSetSelection( aNewSel );
1928 	}
1929 	else
1930 	{
1931         ImpSetSelection( aNewSel );
1932 		ShowSelection( aTmpNewSel );
1933 	}
1934 
1935     sal_Bool bForceCursor =  mpImpl->mpDDInfo ? sal_False : sal_True; // && !mbInSelection
1936     ImpShowCursor( mpImpl->mbAutoScroll, bForceCursor, sal_False );
1937 	return sal_True;
1938 }
1939 
1940 sal_Bool TextView::IsSelectionAtPoint( const Point& rPosPixel )
1941 {
1942 //  if ( !Rectangle( Point(), mpImpl->mpWindow->GetOutputSizePixel() ).IsInside( rPosPixel ) && !mbInSelection )
1943 //		return sal_False;
1944 
1945 	Point aDocPos = GetDocPos( rPosPixel );
1946     TextPaM aPaM = mpImpl->mpTextEngine->GetPaM( aDocPos, sal_False );
1947 	// Bei Hyperlinks D&D auch ohne Selektion starten.
1948 	// BeginDrag wird aber nur gerufen, wenn IsSelectionAtPoint()
1949 	// Problem: IsSelectionAtPoint wird bei Command() nicht gerufen,
1950 	// wenn vorher im MBDown schon sal_False returnt wurde.
1951 	return ( IsInSelection( aPaM ) ||
1952             ( /* mpImpl->mpSelEngine->IsInCommand() && */ mpImpl->mpTextEngine->FindAttrib( aPaM, TEXTATTR_HYPERLINK ) ) );
1953 }
1954 
1955 sal_Bool TextView::IsInSelection( const TextPaM& rPaM )
1956 {
1957     TextSelection aSel = mpImpl->maSelection;
1958 	aSel.Justify();
1959 
1960 	sal_uLong nStartNode = aSel.GetStart().GetPara();
1961 	sal_uLong nEndNode = aSel.GetEnd().GetPara();
1962 	sal_uLong nCurNode = rPaM.GetPara();
1963 
1964 	if ( ( nCurNode > nStartNode ) && ( nCurNode < nEndNode ) )
1965 		return sal_True;
1966 
1967 	if ( nStartNode == nEndNode )
1968 	{
1969 		if ( nCurNode == nStartNode )
1970 			if ( ( rPaM.GetIndex() >= aSel.GetStart().GetIndex() ) && ( rPaM.GetIndex() < aSel.GetEnd().GetIndex() ) )
1971 				return sal_True;
1972 	}
1973 	else if ( ( nCurNode == nStartNode ) && ( rPaM.GetIndex() >= aSel.GetStart().GetIndex() ) )
1974 		return sal_True;
1975 	else if ( ( nCurNode == nEndNode ) && ( rPaM.GetIndex() < aSel.GetEnd().GetIndex() ) )
1976 		return sal_True;
1977 
1978 	return sal_False;
1979 }
1980 
1981 void TextView::ImpHideDDCursor()
1982 {
1983     if ( mpImpl->mpDDInfo && mpImpl->mpDDInfo->mbVisCursor )
1984 	{
1985         mpImpl->mpDDInfo->maCursor.Hide();
1986         mpImpl->mpDDInfo->mbVisCursor = sal_False;
1987 	}
1988 }
1989 
1990 void TextView::ImpShowDDCursor()
1991 {
1992     if ( !mpImpl->mpDDInfo->mbVisCursor )
1993 	{
1994         Rectangle aCursor = mpImpl->mpTextEngine->PaMtoEditCursor( mpImpl->mpDDInfo->maDropPos, sal_True );
1995 		aCursor.Right()++;
1996 		aCursor.SetPos( GetWindowPos( aCursor.TopLeft() ) );
1997 
1998         mpImpl->mpDDInfo->maCursor.SetWindow( mpImpl->mpWindow );
1999         mpImpl->mpDDInfo->maCursor.SetPos( aCursor.TopLeft() );
2000         mpImpl->mpDDInfo->maCursor.SetSize( aCursor.GetSize() );
2001         mpImpl->mpDDInfo->maCursor.Show();
2002         mpImpl->mpDDInfo->mbVisCursor = sal_True;
2003 	}
2004 }
2005 
2006 void TextView::SetPaintSelection( sal_Bool bPaint )
2007 {
2008     if ( bPaint != mpImpl->mbPaintSelection )
2009 	{
2010         mpImpl->mbPaintSelection = bPaint;
2011         ShowSelection( mpImpl->maSelection );
2012 	}
2013 }
2014 
2015 void TextView::SetHighlightSelection( sal_Bool bSelectByHighlight )
2016 {
2017     if ( bSelectByHighlight != mpImpl->mbHighlightSelection )
2018 	{
2019 		// Falls umschalten zwischendurch moeglich...
2020         mpImpl->mbHighlightSelection = bSelectByHighlight;
2021 	}
2022 }
2023 
2024 sal_Bool TextView::Read( SvStream& rInput )
2025 {
2026     sal_Bool bDone = mpImpl->mpTextEngine->Read( rInput, &mpImpl->maSelection );
2027 	ShowCursor();
2028 	return bDone;
2029 }
2030 
2031 sal_Bool TextView::Write( SvStream& rOutput )
2032 {
2033     return mpImpl->mpTextEngine->Read( rOutput, &mpImpl->maSelection );
2034 }
2035 
2036 bool TextView::ImplTruncateNewText( rtl::OUString& rNewText ) const
2037 {
2038 	bool bTruncated = false;
2039 
2040     if( rNewText.getLength() > 65534 ) // limit to String API
2041     {
2042         rNewText = rNewText.copy( 0, 65534 );
2043         bTruncated = true;
2044     }
2045 
2046     sal_uLong nMaxLen = mpImpl->mpTextEngine->GetMaxTextLen();
2047     // 0 means unlimited, there is just the String API limit handled above
2048     if( nMaxLen != 0 )
2049     {
2050         sal_uLong nCurLen = mpImpl->mpTextEngine->GetTextLen();
2051 
2052         sal_uInt32 nNewLen = rNewText.getLength();
2053         if ( nCurLen + nNewLen > nMaxLen )
2054         {
2055             // see how much text will be replaced
2056             sal_uLong nSelLen = mpImpl->mpTextEngine->GetTextLen( mpImpl->maSelection );
2057             if ( nCurLen + nNewLen - nSelLen > nMaxLen )
2058             {
2059                 sal_uInt32 nTruncatedLen = static_cast<sal_uInt32>(nMaxLen - (nCurLen - nSelLen));
2060                 rNewText = rNewText.copy( 0, nTruncatedLen );
2061                 bTruncated = true;
2062             }
2063         }
2064     }
2065 	return bTruncated;
2066 }
2067 
2068 sal_Bool TextView::ImplCheckTextLen( const String& rNewText )
2069 {
2070 	sal_Bool bOK = sal_True;
2071     if ( mpImpl->mpTextEngine->GetMaxTextLen() )
2072 	{
2073         sal_uLong n = mpImpl->mpTextEngine->GetTextLen();
2074 		n += rNewText.Len();
2075         if ( n > mpImpl->mpTextEngine->GetMaxTextLen() )
2076 		{
2077 			// nur dann noch ermitteln, wie viel Text geloescht wird
2078             n -= mpImpl->mpTextEngine->GetTextLen( mpImpl->maSelection );
2079             if ( n > mpImpl->mpTextEngine->GetMaxTextLen() )
2080 			{
2081 				// Beep hat hier eigentlich nichts verloren, sondern lieber ein Hdl,
2082 				// aber so funktioniert es wenigstens in ME, BasicIDE, SourceView
2083 				Sound::Beep();
2084 				bOK = sal_False;
2085 			}
2086 		}
2087 	}
2088 	return bOK;
2089 }
2090 
2091 void TextView::dragGestureRecognized( const ::com::sun::star::datatransfer::dnd::DragGestureEvent& rDGE ) throw (::com::sun::star::uno::RuntimeException)
2092 {
2093     if ( mpImpl->mbClickedInSelection )
2094     {
2095 	    vos::OGuard aVclGuard( Application::GetSolarMutex() );
2096 
2097         DBG_ASSERT( mpImpl->maSelection.HasRange(), "TextView::dragGestureRecognized: mpImpl->mbClickedInSelection, but no selection?" );
2098 
2099         delete mpImpl->mpDDInfo;
2100         mpImpl->mpDDInfo = new TextDDInfo;
2101         mpImpl->mpDDInfo->mbStarterOfDD = sal_True;
2102 
2103 	    TETextDataObject* pDataObj = new TETextDataObject( GetSelected() );
2104 
2105         if ( mpImpl->mpTextEngine->HasAttrib( TEXTATTR_HYPERLINK ) )  // Dann auch als HTML
2106             mpImpl->mpTextEngine->Write( pDataObj->GetHTMLStream(), &mpImpl->maSelection, sal_True );
2107 
2108 
2109         /*
2110 	    // D&D eines Hyperlinks.
2111 	    // Besser waere es im MBDown sich den MBDownPaM zu merken,
2112 	    // ist dann aber inkompatibel => spaeter mal umstellen.
2113         TextPaM aPaM( mpImpl->mpTextEngine->GetPaM( GetDocPos( GetWindow()->GetPointerPosPixel() ) ) );
2114         const TextCharAttrib* pAttr = mpImpl->mpTextEngine->FindCharAttrib( aPaM, TEXTATTR_HYPERLINK );
2115 	    if ( pAttr )
2116 	    {
2117 		    aSel = aPaM;
2118 		    aSel.GetStart().GetIndex() = pAttr->GetStart();
2119 		    aSel.GetEnd().GetIndex() = pAttr->GetEnd();
2120 
2121 		    const TextAttribHyperLink& rLink = (const TextAttribHyperLink&)pAttr->GetAttr();
2122 		    String aText( rLink.GetDescription() );
2123 		    if ( !aText.Len() )
2124                 aText = mpImpl->mpTextEngine->GetText( aSel );
2125 		    INetBookmark aBookmark( rLink.GetURL(), aText );
2126 		    aBookmark.CopyDragServer();
2127 	    }
2128         */
2129 
2130         mpImpl->mpCursor->Hide();
2131 
2132         sal_Int8 nActions = datatransfer::dnd::DNDConstants::ACTION_COPY;
2133         if ( !IsReadOnly() )
2134             nActions |= datatransfer::dnd::DNDConstants::ACTION_MOVE;
2135         rDGE.DragSource->startDrag( rDGE, nActions, 0 /*cursor*/, 0 /*image*/, pDataObj, mpImpl->mxDnDListener );
2136     }
2137 }
2138 
2139 void TextView::dragDropEnd( const ::com::sun::star::datatransfer::dnd::DragSourceDropEvent& ) throw (::com::sun::star::uno::RuntimeException)
2140 {
2141     ImpHideDDCursor();
2142     delete mpImpl->mpDDInfo;
2143     mpImpl->mpDDInfo = NULL;
2144 }
2145 
2146 void TextView::drop( const ::com::sun::star::datatransfer::dnd::DropTargetDropEvent& rDTDE ) throw (::com::sun::star::uno::RuntimeException)
2147 {
2148 	vos::OGuard aVclGuard( Application::GetSolarMutex() );
2149 
2150 	sal_Bool bChanges = sal_False;
2151     if ( !mpImpl->mbReadOnly && mpImpl->mpDDInfo )
2152 	{
2153 		ImpHideDDCursor();
2154 
2155 		// Daten fuer das loeschen nach einem DROP_MOVE:
2156         TextSelection aPrevSel( mpImpl->maSelection );
2157 		aPrevSel.Justify();
2158         sal_uLong nPrevParaCount = mpImpl->mpTextEngine->GetParagraphCount();
2159         sal_uInt16 nPrevStartParaLen = mpImpl->mpTextEngine->GetTextLen( aPrevSel.GetStart().GetPara() );
2160 
2161 		sal_Bool bStarterOfDD = sal_False;
2162         for ( sal_uInt16 nView = mpImpl->mpTextEngine->GetViewCount(); nView && !bStarterOfDD; )
2163             bStarterOfDD = mpImpl->mpTextEngine->GetView( --nView )->mpImpl->mpDDInfo ? mpImpl->mpTextEngine->GetView( nView )->mpImpl->mpDDInfo->mbStarterOfDD : sal_False;
2164 
2165 		HideSelection();
2166         ImpSetSelection( mpImpl->mpDDInfo->maDropPos );
2167 
2168         mpImpl->mpTextEngine->UndoActionStart();
2169 
2170 		String aText;
2171 		uno::Reference< datatransfer::XTransferable > xDataObj = rDTDE.Transferable;
2172 		if ( xDataObj.is() )
2173 		{
2174 		    datatransfer::DataFlavor aFlavor;
2175 		    SotExchange::GetFormatDataFlavor( SOT_FORMAT_STRING, aFlavor );
2176 		    if ( xDataObj->isDataFlavorSupported( aFlavor ) )
2177 		    {
2178 			    uno::Any aData = xDataObj->getTransferData( aFlavor );
2179 			    ::rtl::OUString aOUString;
2180 			    aData >>= aOUString;
2181                 aText = aOUString;
2182 		        aText.ConvertLineEnd( LINEEND_LF );
2183 		    }
2184         }
2185 
2186 		if ( aText.Len() && ( aText.GetChar( aText.Len()-1 ) == LINE_SEP ) )
2187 			aText.Erase( aText.Len()-1 );
2188 
2189 		TextPaM aTempStart = mpImpl->maSelection.GetStart();
2190         if ( ImplCheckTextLen( aText ) )
2191             ImpSetSelection( mpImpl->mpTextEngine->ImpInsertText( mpImpl->mpDDInfo->maDropPos, aText ) );
2192         if(mpImpl->mbSupportProtectAttribute)
2193         {
2194             mpImpl->mpTextEngine->SetAttrib( TextAttribProtect(),
2195                 aTempStart.GetPara(),
2196                 aTempStart.GetIndex(),
2197                 mpImpl->maSelection.GetEnd().GetIndex(), sal_False );
2198         }
2199 
2200         if ( aPrevSel.HasRange() &&
2201                 !mpImpl->mbSupportProtectAttribute && // don't remove currently selected element
2202                 (( rDTDE.DropAction & datatransfer::dnd::DNDConstants::ACTION_MOVE ) || !bStarterOfDD) )
2203 		{
2204 			// ggf. Selection anpasssen:
2205             if ( ( mpImpl->mpDDInfo->maDropPos.GetPara() < aPrevSel.GetStart().GetPara() ) ||
2206                  ( ( mpImpl->mpDDInfo->maDropPos.GetPara() == aPrevSel.GetStart().GetPara() )
2207                         && ( mpImpl->mpDDInfo->maDropPos.GetIndex() < aPrevSel.GetStart().GetIndex() ) ) )
2208 			{
2209 				sal_uLong nNewParasBeforeSelection =
2210                     mpImpl->mpTextEngine->GetParagraphCount() -    nPrevParaCount;
2211 
2212 				aPrevSel.GetStart().GetPara() += nNewParasBeforeSelection;
2213 				aPrevSel.GetEnd().GetPara() += nNewParasBeforeSelection;
2214 
2215                 if ( mpImpl->mpDDInfo->maDropPos.GetPara() == aPrevSel.GetStart().GetPara() )
2216 				{
2217 					sal_uInt16 nNewChars =
2218                         mpImpl->mpTextEngine->GetTextLen( aPrevSel.GetStart().GetPara() ) - nPrevStartParaLen;
2219 
2220 					aPrevSel.GetStart().GetIndex() =
2221                         aPrevSel.GetStart().GetIndex() + nNewChars;
2222 					if ( aPrevSel.GetStart().GetPara() == aPrevSel.GetEnd().GetPara() )
2223 						aPrevSel.GetEnd().GetIndex() =
2224                             aPrevSel.GetEnd().GetIndex() + nNewChars;
2225 				}
2226 			}
2227 			else
2228 			{
2229 				// aktuelle Selektion anpassen
2230                 TextPaM aPaM = mpImpl->maSelection.GetStart();
2231 				aPaM.GetPara() -= ( aPrevSel.GetEnd().GetPara() - aPrevSel.GetStart().GetPara() );
2232                 if ( aPrevSel.GetEnd().GetPara() == mpImpl->mpDDInfo->maDropPos.GetPara() )
2233 				{
2234 					aPaM.GetIndex() =
2235                         aPaM.GetIndex() - aPrevSel.GetEnd().GetIndex();
2236                     if ( aPrevSel.GetStart().GetPara() == mpImpl->mpDDInfo->maDropPos.GetPara() )
2237 						aPaM.GetIndex() =
2238                             aPaM.GetIndex() + aPrevSel.GetStart().GetIndex();
2239 				}
2240 				ImpSetSelection( aPaM );
2241 
2242 			}
2243             mpImpl->mpTextEngine->ImpDeleteText( aPrevSel );
2244 		}
2245 
2246         mpImpl->mpTextEngine->UndoActionEnd();
2247 
2248         delete mpImpl->mpDDInfo;
2249         mpImpl->mpDDInfo = 0;
2250 
2251         mpImpl->mpTextEngine->FormatAndUpdate( this );
2252 
2253         mpImpl->mpTextEngine->Broadcast( TextHint( TEXT_HINT_MODIFIED ) );
2254     }
2255     rDTDE.Context->dropComplete( bChanges );
2256 }
2257 
2258 void TextView::dragEnter( const ::com::sun::star::datatransfer::dnd::DropTargetDragEnterEvent& ) throw (::com::sun::star::uno::RuntimeException)
2259 {
2260 }
2261 
2262 void TextView::dragExit( const ::com::sun::star::datatransfer::dnd::DropTargetEvent& ) throw (::com::sun::star::uno::RuntimeException)
2263 {
2264 	vos::OGuard aVclGuard( Application::GetSolarMutex() );
2265     ImpHideDDCursor();
2266 }
2267 
2268 void TextView::dragOver( const ::com::sun::star::datatransfer::dnd::DropTargetDragEvent& rDTDE ) throw (::com::sun::star::uno::RuntimeException)
2269 {
2270 	vos::OGuard aVclGuard( Application::GetSolarMutex() );
2271 
2272     if ( !mpImpl->mpDDInfo )
2273         mpImpl->mpDDInfo = new TextDDInfo;
2274 
2275     TextPaM aPrevDropPos = mpImpl->mpDDInfo->maDropPos;
2276 	Point aMousePos( rDTDE.LocationX, rDTDE.LocationY );
2277 	Point aDocPos = GetDocPos( aMousePos );
2278     mpImpl->mpDDInfo->maDropPos = mpImpl->mpTextEngine->GetPaM( aDocPos );
2279 
2280 /*
2281     Size aOutSize = mpImpl->mpWindow->GetOutputSizePixel();
2282 	if ( ( aMousePos.X() < 0 ) || ( aMousePos.X() > aOutSize.Width() ) ||
2283 		 ( aMousePos.Y() < 0 ) || ( aMousePos.Y() > aOutSize.Height() ) )
2284 	{
2285 		// Scroll?
2286 		// No, I will not receive events for this...
2287 	}
2288 */
2289 
2290     sal_Bool bProtected = sal_False;
2291     if(mpImpl->mbSupportProtectAttribute)
2292     {
2293         const TextCharAttrib* pStartAttr = mpImpl->mpTextEngine->FindCharAttrib(
2294                     mpImpl->mpDDInfo->maDropPos,
2295                     TEXTATTR_PROTECTED );
2296         bProtected = pStartAttr != 0 &&
2297                 pStartAttr->GetStart() != mpImpl->mpDDInfo->maDropPos.GetIndex() &&
2298                 pStartAttr->GetEnd() != mpImpl->mpDDInfo->maDropPos.GetIndex();
2299     }
2300     // Don't drop in selection or in read only engine
2301     if ( IsReadOnly() || IsInSelection( mpImpl->mpDDInfo->maDropPos ) || bProtected)
2302 	{
2303 		ImpHideDDCursor();
2304         rDTDE.Context->rejectDrag();
2305 	}
2306     else
2307     {
2308 	    // Alten Cursor wegzeichnen...
2309         if ( !mpImpl->mpDDInfo->mbVisCursor || ( aPrevDropPos != mpImpl->mpDDInfo->maDropPos ) )
2310 	    {
2311 		    ImpHideDDCursor();
2312 		    ImpShowDDCursor();
2313 	    }
2314         rDTDE.Context->acceptDrag( rDTDE.DropAction );
2315     }
2316 }
2317 
2318 Point TextView::ImpGetOutputStartPos( const Point& rStartDocPos ) const
2319 {
2320 	Point aStartPos( -rStartDocPos.X(), -rStartDocPos.Y() );
2321     if ( mpImpl->mpTextEngine->IsRightToLeft() )
2322     {
2323         Size aSz = mpImpl->mpWindow->GetOutputSizePixel();
2324 		aStartPos.X() = rStartDocPos.X() + aSz.Width() - 1; // -1: Start is 0
2325     }
2326     return aStartPos;
2327 }
2328 
2329 Point TextView::GetDocPos( const Point& rWindowPos ) const
2330 {
2331 	// Fensterposition => Dokumentposition
2332 
2333 	Point aPoint;
2334 
2335     aPoint.Y() = rWindowPos.Y() + mpImpl->maStartDocPos.Y();
2336 
2337     if ( !mpImpl->mpTextEngine->IsRightToLeft() )
2338     {
2339         aPoint.X() = rWindowPos.X() + mpImpl->maStartDocPos.X();
2340     }
2341     else
2342     {
2343         Size aSz = mpImpl->mpWindow->GetOutputSizePixel();
2344         aPoint.X() = ( aSz.Width() - 1 ) - rWindowPos.X() + mpImpl->maStartDocPos.X();
2345     }
2346 
2347 	return aPoint;
2348 }
2349 
2350 Point TextView::GetWindowPos( const Point& rDocPos ) const
2351 {
2352 	// Dokumentposition => Fensterposition
2353 
2354 	Point aPoint;
2355 
2356     aPoint.Y() = rDocPos.Y() - mpImpl->maStartDocPos.Y();
2357 
2358     if ( !mpImpl->mpTextEngine->IsRightToLeft() )
2359     {
2360         aPoint.X() = rDocPos.X() - mpImpl->maStartDocPos.X();
2361     }
2362     else
2363     {
2364         Size aSz = mpImpl->mpWindow->GetOutputSizePixel();
2365         aPoint.X() = ( aSz.Width() - 1 ) - ( rDocPos.X() - mpImpl->maStartDocPos.X() );
2366     }
2367 
2368 	return aPoint;
2369 }
2370 
2371 sal_Int32 TextView::GetLineNumberOfCursorInSelection() const
2372 {
2373  // PROGRESS
2374     sal_Int32 nLineNo = -1;
2375     if( mpImpl->mbCursorEnabled )
2376     {
2377         TextPaM aPaM = GetSelection().GetEnd();
2378         TEParaPortion* pPPortion = mpImpl->mpTextEngine->mpTEParaPortions->GetObject( aPaM.GetPara() );
2379         nLineNo = pPPortion->GetLineNumber( aPaM.GetIndex(), sal_False );
2380         if( mpImpl->mbCursorAtEndOfLine )
2381             --nLineNo;
2382     }
2383     return nLineNo;
2384 }
2385 
2386 
2387 // -------------------------------------------------------------------------
2388 // (+) class TextSelFunctionSet
2389 // -------------------------------------------------------------------------
2390 TextSelFunctionSet::TextSelFunctionSet( TextView* pView )
2391 {
2392 	mpView = pView;
2393 }
2394 
2395 void __EXPORT TextSelFunctionSet::BeginDrag()
2396 {
2397 }
2398 
2399 void __EXPORT TextSelFunctionSet::CreateAnchor()
2400 {
2401 //	TextSelection aSel( mpView->GetSelection() );
2402 //	aSel.GetStart() = aSel.GetEnd();
2403 //	mpView->SetSelection( aSel );
2404 
2405 	// Es darf kein ShowCursor folgen:
2406 	mpView->HideSelection();
2407     mpView->ImpSetSelection( mpView->mpImpl->maSelection.GetEnd() );
2408 }
2409 
2410 sal_Bool __EXPORT TextSelFunctionSet::SetCursorAtPoint( const Point& rPointPixel, sal_Bool )
2411 {
2412 	return mpView->SetCursorAtPoint( rPointPixel );
2413 }
2414 
2415 sal_Bool __EXPORT TextSelFunctionSet::IsSelectionAtPoint( const Point& rPointPixel )
2416 {
2417 	return mpView->IsSelectionAtPoint( rPointPixel );
2418 }
2419 
2420 void __EXPORT TextSelFunctionSet::DeselectAll()
2421 {
2422 	CreateAnchor();
2423 }
2424 
2425 void __EXPORT TextSelFunctionSet::DeselectAtPoint( const Point& )
2426 {
2427 	// Nur bei Mehrfachselektion
2428 }
2429 
2430 void __EXPORT TextSelFunctionSet::DestroyAnchor()
2431 {
2432 	// Nur bei Mehrfachselektion
2433 }
2434 TextEngine*         TextView::GetTextEngine() const
2435 { return mpImpl->mpTextEngine; }
2436 Window*             TextView::GetWindow() const
2437 { return mpImpl->mpWindow; }
2438 void                TextView::EnableCursor( sal_Bool bEnable )
2439 { mpImpl->mbCursorEnabled = bEnable; }
2440 sal_Bool                TextView::IsCursorEnabled() const
2441 { return mpImpl->mbCursorEnabled; }
2442 void                TextView::SetStartDocPos( const Point& rPos )
2443 { mpImpl->maStartDocPos = rPos; }
2444 const Point&        TextView::GetStartDocPos() const
2445 { return mpImpl->maStartDocPos; }
2446 void                TextView::SetAutoIndentMode( sal_Bool bAutoIndent )
2447 { mpImpl->mbAutoIndent = bAutoIndent; }
2448 sal_Bool                TextView::IsAutoIndentMode() const
2449 { return mpImpl->mbAutoIndent; }
2450 sal_Bool                TextView::IsReadOnly() const
2451 { return mpImpl->mbReadOnly; }
2452 void                TextView::SetAutoScroll( sal_Bool bAutoScroll )
2453 { mpImpl->mbAutoScroll = bAutoScroll; }
2454 sal_Bool                TextView::IsAutoScroll() const
2455 { return mpImpl->mbAutoScroll; }
2456 sal_Bool                TextView::IsPaintSelection() const
2457 { return mpImpl->mbPaintSelection; }
2458 sal_Bool                TextView::IsHighlightSelection() const
2459 { return mpImpl->mbHighlightSelection; }
2460 sal_Bool                TextView::HasSelection() const
2461 { return mpImpl->maSelection.HasRange(); }
2462 sal_Bool                TextView::IsInsertMode() const
2463 { return mpImpl->mbInsertMode; }
2464 void                TextView::SupportProtectAttribute(sal_Bool bSupport)
2465 { mpImpl->mbSupportProtectAttribute = bSupport;}
2466 
2467