xref: /aoo42x/main/editeng/source/uno/unoforou.cxx (revision 190118d0)
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_editeng.hxx"
26 
27 #include <algorithm>
28 #include <svl/style.hxx>
29 #include <com/sun/star/i18n/WordType.hpp>
30 
31 #include <svl/itemset.hxx>
32 #include <editeng/editeng.hxx>
33 #include <editeng/editdata.hxx>
34 #include <editeng/outliner.hxx>
35 #include <editeng/unoedhlp.hxx>
36 #include <svl/poolitem.hxx>
37 #include <vcl/wrkwin.hxx>
38 #include <editeng/eeitem.hxx>
39 
40 #include <editeng/unoforou.hxx>
41 #include <editeng/unofored.hxx>
42 #include <editeng/outlobj.hxx>
43 
44 using namespace ::com::sun::star;
45 
46 //------------------------------------------------------------------------
47 
48 SvxOutlinerForwarder::SvxOutlinerForwarder( Outliner& rOutl, sal_Bool bOutlText /* = sal_False */ ) :
49 	rOutliner( rOutl ),
50     bOutlinerText( bOutlText ),
51     mpAttribsCache( NULL ),
52     mpParaAttribsCache( NULL ),
53     mnParaAttribsCache( 0 )
54 {
55 }
56 
57 SvxOutlinerForwarder::~SvxOutlinerForwarder()
58 {
59 	flushCache();
60 }
61 
62 sal_uInt16 SvxOutlinerForwarder::GetParagraphCount() const
63 {
64 	return (sal_uInt16)rOutliner.GetParagraphCount();
65 }
66 
67 sal_uInt16 SvxOutlinerForwarder::GetTextLen( sal_uInt16 nParagraph ) const
68 {
69 	return rOutliner.GetEditEngine().GetTextLen( nParagraph );
70 }
71 
72 String SvxOutlinerForwarder::GetText( const ESelection& rSel ) const
73 {
74 	//!	GetText(ESelection) sollte es wohl auch mal am Outliner geben
75 	//	solange den Hack fuer die EditEngine uebernehmen:
76 	EditEngine* pEditEngine = (EditEngine*)&rOutliner.GetEditEngine();
77 	return pEditEngine->GetText( rSel, LINEEND_LF );
78 }
79 
80 static SfxItemSet ImplOutlinerForwarderGetAttribs( const ESelection& rSel, sal_Bool bOnlyHardAttrib, EditEngine& rEditEngine )
81 {
82 	if( rSel.nStartPara == rSel.nEndPara )
83 	{
84 		sal_uInt8 nFlags = 0;
85 
86 		switch( bOnlyHardAttrib )
87 		{
88 		case EditEngineAttribs_All:
89 			nFlags = GETATTRIBS_ALL;
90 			break;
91 		case EditEngineAttribs_HardAndPara:
92 			nFlags = GETATTRIBS_PARAATTRIBS|GETATTRIBS_CHARATTRIBS;
93 			break;
94 		case EditEngineAttribs_OnlyHard:
95 			nFlags = GETATTRIBS_CHARATTRIBS;
96 			break;
97 		default:
98 			DBG_ERROR("unknown flags for SvxOutlinerForwarder::GetAttribs");
99 		}
100 		return rEditEngine.GetAttribs( rSel.nStartPara, rSel.nStartPos, rSel.nEndPos, nFlags );
101 	}
102 	else
103 	{
104 		return rEditEngine.GetAttribs( rSel, bOnlyHardAttrib );
105 	}
106 }
107 
108 SfxItemSet SvxOutlinerForwarder::GetAttribs( const ESelection& rSel, sal_Bool bOnlyHardAttrib ) const
109 {
110 	if( mpAttribsCache && ( 0 == bOnlyHardAttrib ) )
111 	{
112 		// have we the correct set in cache?
113 		if( ((SvxOutlinerForwarder*)this)->maAttribCacheSelection.IsEqual(rSel) )
114 		{
115 			// yes! just return the cache
116 			return *mpAttribsCache;
117 		}
118 		else
119 		{
120 			// no, we need delete the old cache
121 			delete mpAttribsCache;
122 			mpAttribsCache = NULL;
123 		}
124 	}
125 
126 	//!	gibt's das nicht am Outliner ???
127 	//!	und warum ist GetAttribs an der EditEngine nicht const?
128 	EditEngine& rEditEngine = (EditEngine&)rOutliner.GetEditEngine();
129 
130 	SfxItemSet aSet( ImplOutlinerForwarderGetAttribs( rSel, bOnlyHardAttrib, rEditEngine ) );
131 
132 	if( 0 == bOnlyHardAttrib )
133 	{
134 		mpAttribsCache = new SfxItemSet( aSet );
135 		maAttribCacheSelection = rSel;
136 	}
137 
138 	SfxStyleSheet* pStyle = rEditEngine.GetStyleSheet( rSel.nStartPara );
139 	if( pStyle )
140 		aSet.SetParent( &(pStyle->GetItemSet() ) );
141 
142 	return aSet;
143 }
144 
145 SfxItemSet SvxOutlinerForwarder::GetParaAttribs( sal_uInt16 nPara ) const
146 {
147 	if( mpParaAttribsCache )
148 	{
149 		// have we the correct set in cache?
150 		if( nPara == mnParaAttribsCache )
151 		{
152 			// yes! just return the cache
153 			return *mpParaAttribsCache;
154 		}
155 		else
156 		{
157 			// no, we need delete the old cache
158 			delete mpParaAttribsCache;
159 			mpParaAttribsCache = NULL;
160 		}
161 	}
162 
163 	mpParaAttribsCache = new SfxItemSet( rOutliner.GetParaAttribs( nPara ) );
164 	mnParaAttribsCache = nPara;
165 
166 	EditEngine& rEditEngine = (EditEngine&)rOutliner.GetEditEngine();
167 
168 	SfxStyleSheet* pStyle = rEditEngine.GetStyleSheet( nPara );
169 	if( pStyle )
170 		mpParaAttribsCache->SetParent( &(pStyle->GetItemSet() ) );
171 
172 	return *mpParaAttribsCache;
173 }
174 
175 void SvxOutlinerForwarder::SetParaAttribs( sal_uInt16 nPara, const SfxItemSet& rSet )
176 {
177 	flushCache();
178 
179 	const SfxItemSet* pOldParent = rSet.GetParent();
180 	if( pOldParent )
181 		((SfxItemSet*)&rSet)->SetParent( NULL );
182 
183 	rOutliner.SetParaAttribs( nPara, rSet );
184 
185 	if( pOldParent )
186 		((SfxItemSet*)&rSet)->SetParent( pOldParent );
187 }
188 
189 void SvxOutlinerForwarder::RemoveAttribs( const ESelection& rSelection, sal_Bool bRemoveParaAttribs, sal_uInt16 nWhich )
190 {
191     rOutliner.RemoveAttribs( rSelection, bRemoveParaAttribs, nWhich );
192 }
193 
194 SfxItemPool* SvxOutlinerForwarder::GetPool() const
195 {
196 	return rOutliner.GetEmptyItemSet().GetPool();
197 }
198 
199 void SvxOutlinerForwarder::GetPortions( sal_uInt16 nPara, SvUShorts& rList ) const
200 {
201 	((EditEngine&)rOutliner.GetEditEngine()).GetPortions( nPara, rList );
202 }
203 
204 void SvxOutlinerForwarder::QuickInsertText( const String& rText, const ESelection& rSel )
205 {
206 	flushCache();
207 	if( rText.Len() == 0 )
208 	{
209 		rOutliner.QuickDelete( rSel );
210 	}
211 	else
212 	{
213 		rOutliner.QuickInsertText( rText, rSel );
214 	}
215 }
216 
217 void SvxOutlinerForwarder::QuickInsertLineBreak( const ESelection& rSel )
218 {
219 	flushCache();
220 	rOutliner.QuickInsertLineBreak( rSel );
221 }
222 
223 void SvxOutlinerForwarder::QuickInsertField( const SvxFieldItem& rFld, const ESelection& rSel )
224 {
225 	flushCache();
226 	rOutliner.QuickInsertField( rFld, rSel );
227 }
228 
229 void SvxOutlinerForwarder::QuickSetAttribs( const SfxItemSet& rSet, const ESelection& rSel )
230 {
231 	flushCache();
232 	rOutliner.QuickSetAttribs( rSet, rSel );
233 }
234 
235 XubString SvxOutlinerForwarder::CalcFieldValue( const SvxFieldItem& rField, sal_uInt16 nPara, sal_uInt16 nPos, Color*& rpTxtColor, Color*& rpFldColor )
236 {
237 	return rOutliner.CalcFieldValue( rField, nPara, nPos, rpTxtColor, rpFldColor );
238 }
239 
240 void SvxOutlinerForwarder::FieldClicked( const SvxFieldItem& rField, sal_uInt16 nPara, xub_StrLen nPos )
241 {
242     rOutliner.FieldClicked( rField, nPara, nPos );
243 }
244 
245 sal_Bool SvxOutlinerForwarder::IsValid() const
246 {
247     // cannot reliably query outliner state
248     // while in the middle of an update
249     return rOutliner.GetUpdateMode();
250 }
251 
252 extern sal_uInt16 GetSvxEditEngineItemState( EditEngine& rEditEngine, const ESelection& rSel, sal_uInt16 nWhich );
253 
254 sal_uInt16 SvxOutlinerForwarder::GetItemState( const ESelection& rSel, sal_uInt16 nWhich ) const
255 {
256 	return GetSvxEditEngineItemState( (EditEngine&)rOutliner.GetEditEngine(), rSel, nWhich );
257 }
258 
259 sal_uInt16 SvxOutlinerForwarder::GetItemState( sal_uInt16 nPara, sal_uInt16 nWhich ) const
260 {
261 	const SfxItemSet& rSet = rOutliner.GetParaAttribs( nPara );
262 	return rSet.GetItemState( nWhich );
263 }
264 
265 
266 void SvxOutlinerForwarder::flushCache()
267 {
268 	if( mpAttribsCache )
269 	{
270 		delete mpAttribsCache;
271 		mpAttribsCache = NULL;
272 	}
273 
274 	if( mpParaAttribsCache )
275 	{
276 		delete mpParaAttribsCache;
277 		mpParaAttribsCache = NULL;
278 	}
279 }
280 
281 LanguageType SvxOutlinerForwarder::GetLanguage( sal_uInt16 nPara, sal_uInt16 nIndex ) const
282 {
283     return rOutliner.GetLanguage(nPara, nIndex);
284 }
285 
286 sal_uInt16 SvxOutlinerForwarder::GetFieldCount( sal_uInt16 nPara ) const
287 {
288     return rOutliner.GetEditEngine().GetFieldCount(nPara);
289 }
290 
291 EFieldInfo SvxOutlinerForwarder::GetFieldInfo( sal_uInt16 nPara, sal_uInt16 nField ) const
292 {
293     return rOutliner.GetEditEngine().GetFieldInfo( nPara, nField );
294 }
295 
296 EBulletInfo SvxOutlinerForwarder::GetBulletInfo( sal_uInt16 nPara ) const
297 {
298     return rOutliner.GetBulletInfo( nPara );
299 }
300 
301 Rectangle SvxOutlinerForwarder::GetCharBounds( sal_uInt16 nPara, sal_uInt16 nIndex ) const
302 {
303     // #101701#
304     // EditEngine's 'internal' methods like GetCharacterBounds()
305     // don't rotate for vertical text.
306     Size aSize( rOutliner.CalcTextSize() );
307     ::std::swap( aSize.Width(), aSize.Height() );
308     bool bIsVertical( rOutliner.IsVertical() == sal_True );
309 
310     // #108900# Handle virtual position one-past-the end of the string
311     if( nIndex >= GetTextLen(nPara) )
312     {
313         Rectangle aLast;
314 
315         if( nIndex )
316         {
317             // use last character, if possible
318             aLast = rOutliner.GetEditEngine().GetCharacterBounds( EPosition(nPara, nIndex-1) );
319 
320             // move at end of this last character, make one pixel wide
321             aLast.Move( aLast.Right() - aLast.Left(), 0 );
322             aLast.SetSize( Size(1, aLast.GetHeight()) );
323 
324             // take care for CTL
325             aLast = SvxEditSourceHelper::EEToUserSpace( aLast, aSize, bIsVertical );
326         }
327         else
328         {
329             // #109864# Bounds must lie within the paragraph
330             aLast = GetParaBounds( nPara );
331 
332             // #109151# Don't use paragraph height, but line height
333             // instead. aLast is already CTL-correct
334             if( bIsVertical)
335                 aLast.SetSize( Size( rOutliner.GetLineHeight(nPara,0), 1 ) );
336             else
337                 aLast.SetSize( Size( 1, rOutliner.GetLineHeight(nPara,0) ) );
338         }
339 
340         return aLast;
341     }
342     else
343     {
344         return SvxEditSourceHelper::EEToUserSpace( rOutliner.GetEditEngine().GetCharacterBounds( EPosition(nPara, nIndex) ),
345                                                    aSize, bIsVertical );
346     }
347 }
348 
349 Rectangle SvxOutlinerForwarder::GetParaBounds( sal_uInt16 nPara ) const
350 {
351     Point aPnt = rOutliner.GetDocPosTopLeft( nPara );
352     Size aSize = rOutliner.CalcTextSize();
353 
354     if( rOutliner.IsVertical() )
355     {
356         // #101701#
357         // Hargl. Outliner's 'external' methods return the rotated
358         // dimensions, 'internal' methods like GetTextHeight( n )
359         // don't rotate.
360         sal_uLong nWidth = rOutliner.GetTextHeight( nPara );
361 
362         return Rectangle( aSize.Width() - aPnt.Y() - nWidth, 0, aSize.Width() - aPnt.Y(), aSize.Height() );
363     }
364     else
365     {
366         sal_uLong nHeight = rOutliner.GetTextHeight( nPara );
367 
368         return Rectangle( 0, aPnt.Y(), aSize.Width(), aPnt.Y() + nHeight );
369     }
370 }
371 
372 MapMode SvxOutlinerForwarder::GetMapMode() const
373 {
374     return rOutliner.GetRefMapMode();
375 }
376 
377 OutputDevice* SvxOutlinerForwarder::GetRefDevice() const
378 {
379     return rOutliner.GetRefDevice();
380 }
381 
382 sal_Bool SvxOutlinerForwarder::GetIndexAtPoint( const Point& rPos, sal_uInt16& nPara, sal_uInt16& nIndex ) const
383 {
384     // #101701#
385     Size aSize( rOutliner.CalcTextSize() );
386     ::std::swap( aSize.Width(), aSize.Height() );
387     Point aEEPos( SvxEditSourceHelper::UserSpaceToEE( rPos,
388                                                       aSize,
389                                                       rOutliner.IsVertical() == sal_True ));
390 
391     EPosition aDocPos = rOutliner.GetEditEngine().FindDocPosition( aEEPos );
392 
393     nPara = aDocPos.nPara;
394     nIndex = aDocPos.nIndex;
395 
396     return sal_True;
397 }
398 
399 sal_Bool SvxOutlinerForwarder::GetWordIndices( sal_uInt16 nPara, sal_uInt16 nIndex, sal_uInt16& nStart, sal_uInt16& nEnd ) const
400 {
401     ESelection aRes = rOutliner.GetEditEngine().GetWord( ESelection(nPara, nIndex, nPara, nIndex), com::sun::star::i18n::WordType::DICTIONARY_WORD );
402 
403     if( aRes.nStartPara == nPara &&
404         aRes.nStartPara == aRes.nEndPara )
405     {
406         nStart = aRes.nStartPos;
407         nEnd = aRes.nEndPos;
408 
409         return sal_True;
410     }
411 
412     return sal_False;
413 }
414 
415 sal_Bool SvxOutlinerForwarder::GetAttributeRun( sal_uInt16& nStartIndex, sal_uInt16& nEndIndex, sal_uInt16 nPara, sal_uInt16 nIndex ) const
416 {
417     return SvxEditSourceHelper::GetAttributeRun( nStartIndex, nEndIndex, rOutliner.GetEditEngine(), nPara, nIndex );
418 }
419 
420 sal_uInt16 SvxOutlinerForwarder::GetLineCount( sal_uInt16 nPara ) const
421 {
422     return static_cast < sal_uInt16 >( rOutliner.GetLineCount(nPara) );
423 }
424 
425 sal_uInt16 SvxOutlinerForwarder::GetLineLen( sal_uInt16 nPara, sal_uInt16 nLine ) const
426 {
427     return rOutliner.GetLineLen(nPara, nLine);
428 }
429 
430 void SvxOutlinerForwarder::GetLineBoundaries( /*out*/sal_uInt16 &rStart, /*out*/sal_uInt16 &rEnd, sal_uInt16 nPara, sal_uInt16 nLine ) const
431 {
432     return rOutliner.GetEditEngine().GetLineBoundaries( rStart, rEnd, nPara, nLine );
433 }
434 
435 sal_uInt16 SvxOutlinerForwarder::GetLineNumberAtIndex( sal_uInt16 nPara, sal_uInt16 nIndex ) const
436 {
437     return rOutliner.GetEditEngine().GetLineNumberAtIndex( nPara, nIndex );
438 }
439 
440 sal_Bool SvxOutlinerForwarder::QuickFormatDoc( sal_Bool )
441 {
442     rOutliner.QuickFormatDoc();
443 
444     return sal_True;
445 }
446 
447 sal_Bool SvxOutlinerForwarder::Delete( const ESelection& rSelection )
448 {
449 	flushCache();
450     rOutliner.QuickDelete( rSelection );
451     rOutliner.QuickFormatDoc();
452 
453     return sal_True;
454 }
455 
456 sal_Bool SvxOutlinerForwarder::InsertText( const String& rStr, const ESelection& rSelection )
457 {
458 	flushCache();
459     rOutliner.QuickInsertText( rStr, rSelection );
460     rOutliner.QuickFormatDoc();
461 
462     return sal_True;
463 }
464 
465 sal_Int16 SvxOutlinerForwarder::GetDepth( sal_uInt16 nPara ) const
466 {
467     DBG_ASSERT( nPara < GetParagraphCount(), "SvxOutlinerForwarder::GetDepth: Invalid paragraph index");
468 
469     Paragraph* pPara = rOutliner.GetParagraph( nPara );
470 
471     sal_Int16 nLevel = -1;
472 
473     if( pPara )
474         nLevel = rOutliner.GetDepth( nPara );
475 
476     return nLevel;
477 }
478 
479 sal_Bool SvxOutlinerForwarder::SetDepth( sal_uInt16 nPara, sal_Int16 nNewDepth )
480 {
481     DBG_ASSERT( nPara < GetParagraphCount(), "SvxOutlinerForwarder::SetDepth: Invalid paragraph index");
482 
483     if( (nNewDepth >= -1) && (nNewDepth <= 9) && (nPara < GetParagraphCount()) )
484     {
485         Paragraph* pPara = rOutliner.GetParagraph( nPara );
486         if( pPara )
487         {
488             rOutliner.SetDepth( pPara, nNewDepth );
489 
490 //			const bool bOutlinerText = pSdrObject && (pSdrObject->GetObjInventor() == SdrInventor) && (pSdrObject->GetObjIdentifier() == OBJ_OUTLINETEXT);
491             if( bOutlinerText )
492                 rOutliner.SetLevelDependendStyleSheet( nPara );
493 
494             return sal_True;
495         }
496     }
497 
498     return sal_False;
499 }
500 
501 sal_Int16 SvxOutlinerForwarder::GetNumberingStartValue( sal_uInt16 nPara )
502 {
503     if( nPara < GetParagraphCount() )
504     {
505         return rOutliner.GetNumberingStartValue( nPara );
506     }
507     else
508     {
509         DBG_ERROR( "SvxOutlinerForwarder::GetNumberingStartValue)(), Invalid paragraph index");
510         return -1;
511     }
512 }
513 
514 void SvxOutlinerForwarder::SetNumberingStartValue(  sal_uInt16 nPara, sal_Int16 nNumberingStartValue )
515 {
516     if( nPara < GetParagraphCount() )
517     {
518         rOutliner.SetNumberingStartValue( nPara, nNumberingStartValue );
519     }
520     else
521     {
522         DBG_ERROR( "SvxOutlinerForwarder::SetNumberingStartValue)(), Invalid paragraph index");
523     }
524 }
525 
526 sal_Bool SvxOutlinerForwarder::IsParaIsNumberingRestart( sal_uInt16 nPara )
527 {
528     if( nPara < GetParagraphCount() )
529     {
530         return rOutliner.IsParaIsNumberingRestart( nPara );
531     }
532     else
533     {
534         DBG_ERROR( "SvxOutlinerForwarder::IsParaIsNumberingRestart)(), Invalid paragraph index");
535         return sal_False;
536     }
537 }
538 
539 void SvxOutlinerForwarder::SetParaIsNumberingRestart(  sal_uInt16 nPara, sal_Bool bParaIsNumberingRestart )
540 {
541     if( nPara < GetParagraphCount() )
542     {
543         rOutliner.SetParaIsNumberingRestart( nPara, bParaIsNumberingRestart );
544     }
545     else
546     {
547         DBG_ERROR( "SvxOutlinerForwarder::SetParaIsNumberingRestart)(), Invalid paragraph index");
548     }
549 }
550 
551 const SfxItemSet * SvxOutlinerForwarder::GetEmptyItemSetPtr()
552 {
553     EditEngine& rEditEngine = const_cast< EditEngine& >( rOutliner.GetEditEngine() );
554     return &rEditEngine.GetEmptyItemSet();
555 }
556 
557 void SvxOutlinerForwarder::AppendParagraph()
558 {
559     EditEngine& rEditEngine = const_cast< EditEngine& >( rOutliner.GetEditEngine() );
560     rEditEngine.InsertParagraph( rEditEngine.GetParagraphCount(), String::EmptyString() );
561 }
562 
563 xub_StrLen SvxOutlinerForwarder::AppendTextPortion( sal_uInt16 nPara, const String &rText, const SfxItemSet & /*rSet*/ )
564 {
565     xub_StrLen nLen = 0;
566 
567     EditEngine& rEditEngine = const_cast< EditEngine& >( rOutliner.GetEditEngine() );
568     sal_uInt16 nParaCount = rEditEngine.GetParagraphCount();
569     DBG_ASSERT( nPara < nParaCount, "paragraph index out of bounds" );
570     if (/*0 <= nPara && */nPara < nParaCount)
571     {
572         nLen = rEditEngine.GetTextLen( nPara );
573         rEditEngine.QuickInsertText( rText, ESelection( nPara, nLen, nPara, nLen ) );
574     }
575 
576     return nLen;
577 }
578 
579 void  SvxOutlinerForwarder::CopyText(const SvxTextForwarder& rSource)
580 {
581     const SvxOutlinerForwarder* pSourceForwarder = dynamic_cast< const SvxOutlinerForwarder* >( &rSource );
582     if( !pSourceForwarder )
583         return;
584     OutlinerParaObject* pNewOutlinerParaObject = pSourceForwarder->rOutliner.CreateParaObject();
585     rOutliner.SetText( *pNewOutlinerParaObject );
586     delete pNewOutlinerParaObject;
587 }
588 
589 //------------------------------------------------------------------------
590 
591 
592 sal_Int16 SvxTextForwarder::GetNumberingStartValue( sal_uInt16 )
593 {
594     return -1;
595 }
596 
597 void SvxTextForwarder::SetNumberingStartValue( sal_uInt16, sal_Int16 )
598 {
599 }
600 
601 sal_Bool SvxTextForwarder::IsParaIsNumberingRestart( sal_uInt16  )
602 {
603     return sal_False;
604 }
605 
606 void SvxTextForwarder::SetParaIsNumberingRestart( sal_uInt16, sal_Bool )
607 {
608 }
609 
610 //------------------------------------------------------------------------
611 
612