xref: /aoo4110/main/sw/source/core/crsr/callnk.cxx (revision b1cdbd2c)
1*b1cdbd2cSJim Jagielski /**************************************************************
2*b1cdbd2cSJim Jagielski  *
3*b1cdbd2cSJim Jagielski  * Licensed to the Apache Software Foundation (ASF) under one
4*b1cdbd2cSJim Jagielski  * or more contributor license agreements.  See the NOTICE file
5*b1cdbd2cSJim Jagielski  * distributed with this work for additional information
6*b1cdbd2cSJim Jagielski  * regarding copyright ownership.  The ASF licenses this file
7*b1cdbd2cSJim Jagielski  * to you under the Apache License, Version 2.0 (the
8*b1cdbd2cSJim Jagielski  * "License"); you may not use this file except in compliance
9*b1cdbd2cSJim Jagielski  * with the License.  You may obtain a copy of the License at
10*b1cdbd2cSJim Jagielski  *
11*b1cdbd2cSJim Jagielski  *   http://www.apache.org/licenses/LICENSE-2.0
12*b1cdbd2cSJim Jagielski  *
13*b1cdbd2cSJim Jagielski  * Unless required by applicable law or agreed to in writing,
14*b1cdbd2cSJim Jagielski  * software distributed under the License is distributed on an
15*b1cdbd2cSJim Jagielski  * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
16*b1cdbd2cSJim Jagielski  * KIND, either express or implied.  See the License for the
17*b1cdbd2cSJim Jagielski  * specific language governing permissions and limitations
18*b1cdbd2cSJim Jagielski  * under the License.
19*b1cdbd2cSJim Jagielski  *
20*b1cdbd2cSJim Jagielski  *************************************************************/
21*b1cdbd2cSJim Jagielski 
22*b1cdbd2cSJim Jagielski 
23*b1cdbd2cSJim Jagielski 
24*b1cdbd2cSJim Jagielski // MARKER(update_precomp.py): autogen include statement, do not remove
25*b1cdbd2cSJim Jagielski #include "precompiled_sw.hxx"
26*b1cdbd2cSJim Jagielski 
27*b1cdbd2cSJim Jagielski 
28*b1cdbd2cSJim Jagielski #include <hintids.hxx>
29*b1cdbd2cSJim Jagielski 
30*b1cdbd2cSJim Jagielski #ifndef _COM_SUN_STAR_I18N_SCRIPTTYPE_HDL_
31*b1cdbd2cSJim Jagielski #include <com/sun/star/i18n/ScriptType.hdl>
32*b1cdbd2cSJim Jagielski #endif
33*b1cdbd2cSJim Jagielski #include <fmtcntnt.hxx>
34*b1cdbd2cSJim Jagielski #include <txatbase.hxx>
35*b1cdbd2cSJim Jagielski #include <frmatr.hxx>
36*b1cdbd2cSJim Jagielski #include <viscrs.hxx>
37*b1cdbd2cSJim Jagielski #include <callnk.hxx>
38*b1cdbd2cSJim Jagielski #include <crsrsh.hxx>
39*b1cdbd2cSJim Jagielski #include <doc.hxx>
40*b1cdbd2cSJim Jagielski #include <frmfmt.hxx>
41*b1cdbd2cSJim Jagielski #include <txtfrm.hxx>
42*b1cdbd2cSJim Jagielski #include <ndtxt.hxx>
43*b1cdbd2cSJim Jagielski #include <flyfrm.hxx>
44*b1cdbd2cSJim Jagielski #include <breakit.hxx>
45*b1cdbd2cSJim Jagielski 
46*b1cdbd2cSJim Jagielski 
SwCallLink(SwCrsrShell & rSh,sal_uLong nAktNode,xub_StrLen nAktCntnt,sal_uInt8 nAktNdTyp,long nLRPos,bool bAktSelection)47*b1cdbd2cSJim Jagielski SwCallLink::SwCallLink( SwCrsrShell & rSh, sal_uLong nAktNode, xub_StrLen nAktCntnt,
48*b1cdbd2cSJim Jagielski 						sal_uInt8 nAktNdTyp, long nLRPos, bool bAktSelection )
49*b1cdbd2cSJim Jagielski 	: rShell( rSh ), nNode( nAktNode ), nCntnt( nAktCntnt ),
50*b1cdbd2cSJim Jagielski       nNdTyp( nAktNdTyp ), nLeftFrmPos( nLRPos ),
51*b1cdbd2cSJim Jagielski       bHasSelection( bAktSelection )
52*b1cdbd2cSJim Jagielski {
53*b1cdbd2cSJim Jagielski }
54*b1cdbd2cSJim Jagielski 
55*b1cdbd2cSJim Jagielski 
SwCallLink(SwCrsrShell & rSh)56*b1cdbd2cSJim Jagielski SwCallLink::SwCallLink( SwCrsrShell & rSh )
57*b1cdbd2cSJim Jagielski 	: rShell( rSh )
58*b1cdbd2cSJim Jagielski {
59*b1cdbd2cSJim Jagielski 	// SPoint-Werte vom aktuellen Cursor merken
60*b1cdbd2cSJim Jagielski 	SwPaM* pCrsr = rShell.IsTableMode() ? rShell.GetTblCrs() : rShell.GetCrsr();
61*b1cdbd2cSJim Jagielski 	SwNode& rNd = pCrsr->GetPoint()->nNode.GetNode();
62*b1cdbd2cSJim Jagielski 	nNode = rNd.GetIndex();
63*b1cdbd2cSJim Jagielski 	nCntnt = pCrsr->GetPoint()->nContent.GetIndex();
64*b1cdbd2cSJim Jagielski 	nNdTyp = rNd.GetNodeType();
65*b1cdbd2cSJim Jagielski     bHasSelection = ( *pCrsr->GetPoint() != *pCrsr->GetMark() );
66*b1cdbd2cSJim Jagielski 
67*b1cdbd2cSJim Jagielski 	if( ND_TEXTNODE & nNdTyp )
68*b1cdbd2cSJim Jagielski 		nLeftFrmPos = SwCallLink::getLayoutFrm( rShell.GetLayout(), (SwTxtNode&)rNd, nCntnt,
69*b1cdbd2cSJim Jagielski 											!rShell.ActionPend() );
70*b1cdbd2cSJim Jagielski 	else
71*b1cdbd2cSJim Jagielski 	{
72*b1cdbd2cSJim Jagielski 		nLeftFrmPos = 0;
73*b1cdbd2cSJim Jagielski 
74*b1cdbd2cSJim Jagielski 		// eine Sonderbehandlung fuer die SwFeShell: diese setzt beim Loeschen
75*b1cdbd2cSJim Jagielski 		// der Kopf-/Fusszeile, Fussnoten den Cursor auf NULL (Node + Content)
76*b1cdbd2cSJim Jagielski 		// steht der Cursor auf keinem CntntNode, wird sich das im NdType
77*b1cdbd2cSJim Jagielski 		// gespeichert.
78*b1cdbd2cSJim Jagielski 		if( ND_CONTENTNODE & nNdTyp )
79*b1cdbd2cSJim Jagielski 			nNdTyp = 0;
80*b1cdbd2cSJim Jagielski 	}
81*b1cdbd2cSJim Jagielski }
82*b1cdbd2cSJim Jagielski 
83*b1cdbd2cSJim Jagielski 
~SwCallLink()84*b1cdbd2cSJim Jagielski SwCallLink::~SwCallLink()
85*b1cdbd2cSJim Jagielski {
86*b1cdbd2cSJim Jagielski 	if( !nNdTyp || !rShell.bCallChgLnk )		// siehe ctor
87*b1cdbd2cSJim Jagielski 		return ;
88*b1cdbd2cSJim Jagielski 
89*b1cdbd2cSJim Jagielski 	// wird ueber Nodes getravellt, Formate ueberpruefen und im neuen
90*b1cdbd2cSJim Jagielski 	// Node wieder anmelden
91*b1cdbd2cSJim Jagielski 	SwPaM* pCurCrsr = rShell.IsTableMode() ? rShell.GetTblCrs() : rShell.GetCrsr();
92*b1cdbd2cSJim Jagielski 	SwCntntNode * pCNd = pCurCrsr->GetCntntNode();
93*b1cdbd2cSJim Jagielski 	if( !pCNd )
94*b1cdbd2cSJim Jagielski 		return;
95*b1cdbd2cSJim Jagielski 
96*b1cdbd2cSJim Jagielski 	xub_StrLen nCmp, nAktCntnt = pCurCrsr->GetPoint()->nContent.GetIndex();
97*b1cdbd2cSJim Jagielski 	sal_uInt16 nNdWhich = pCNd->GetNodeType();
98*b1cdbd2cSJim Jagielski 	sal_uLong nAktNode = pCurCrsr->GetPoint()->nNode.GetIndex();
99*b1cdbd2cSJim Jagielski 
100*b1cdbd2cSJim Jagielski 	// melde die Shell beim akt. Node als abhaengig an, dadurch koennen
101*b1cdbd2cSJim Jagielski 	// alle Attribut-Aenderungen ueber den Link weiter gemeldet werden.
102*b1cdbd2cSJim Jagielski 	pCNd->Add( &rShell );
103*b1cdbd2cSJim Jagielski 
104*b1cdbd2cSJim Jagielski 	if( nNdTyp != nNdWhich || nNode != nAktNode )
105*b1cdbd2cSJim Jagielski 	{
106*b1cdbd2cSJim Jagielski 		/* immer, wenn zwischen Nodes gesprungen wird, kann es
107*b1cdbd2cSJim Jagielski 		 * vorkommen, das neue Attribute gelten; die Text-Attribute.
108*b1cdbd2cSJim Jagielski 		 * Es muesste also festgestellt werden, welche Attribute
109*b1cdbd2cSJim Jagielski 		 * jetzt gelten; das kann auch gleich der Handler machen
110*b1cdbd2cSJim Jagielski 		 */
111*b1cdbd2cSJim Jagielski 		rShell.CallChgLnk();
112*b1cdbd2cSJim Jagielski 	}
113*b1cdbd2cSJim Jagielski     else if( !bHasSelection != !(*pCurCrsr->GetPoint() != *pCurCrsr->GetMark()) )
114*b1cdbd2cSJim Jagielski     {
115*b1cdbd2cSJim Jagielski         // always call change link when selection changes
116*b1cdbd2cSJim Jagielski 		rShell.CallChgLnk();
117*b1cdbd2cSJim Jagielski     }
118*b1cdbd2cSJim Jagielski 	else if( rShell.aChgLnk.IsSet() && ND_TEXTNODE == nNdWhich &&
119*b1cdbd2cSJim Jagielski 			 nCntnt != nAktCntnt )
120*b1cdbd2cSJim Jagielski 	{
121*b1cdbd2cSJim Jagielski 		// nur wenn mit Left/right getravellt, dann Text-Hints pruefen
122*b1cdbd2cSJim Jagielski 		// und sich nicht der Frame geaendert hat (Spalten!)
123*b1cdbd2cSJim Jagielski 		if( nLeftFrmPos == SwCallLink::getLayoutFrm( rShell.GetLayout(), (SwTxtNode&)*pCNd, nAktCntnt,
124*b1cdbd2cSJim Jagielski 													!rShell.ActionPend() ) &&
125*b1cdbd2cSJim Jagielski 			(( nCmp = nCntnt ) + 1 == nAktCntnt ||			// Right
126*b1cdbd2cSJim Jagielski 			nCntnt -1 == ( nCmp = nAktCntnt )) )			// Left
127*b1cdbd2cSJim Jagielski 		{
128*b1cdbd2cSJim Jagielski 			if( nCmp == nAktCntnt && pCurCrsr->HasMark() ) // left & Sele
129*b1cdbd2cSJim Jagielski 				++nCmp;
130*b1cdbd2cSJim Jagielski 			if ( ((SwTxtNode*)pCNd)->HasHints() )
131*b1cdbd2cSJim Jagielski 			{
132*b1cdbd2cSJim Jagielski 
133*b1cdbd2cSJim Jagielski 				const SwpHints &rHts = ((SwTxtNode*)pCNd)->GetSwpHints();
134*b1cdbd2cSJim Jagielski 				sal_uInt16 n;
135*b1cdbd2cSJim Jagielski 				xub_StrLen nStart;
136*b1cdbd2cSJim Jagielski 				const xub_StrLen *pEnd;
137*b1cdbd2cSJim Jagielski 
138*b1cdbd2cSJim Jagielski 				for( n = 0; n < rHts.Count(); n++ )
139*b1cdbd2cSJim Jagielski 				{
140*b1cdbd2cSJim Jagielski 					const SwTxtAttr* pHt = rHts[ n ];
141*b1cdbd2cSJim Jagielski 					pEnd = pHt->End();
142*b1cdbd2cSJim Jagielski 					nStart = *pHt->GetStart();
143*b1cdbd2cSJim Jagielski 
144*b1cdbd2cSJim Jagielski 					// nur Start oder Start und Ende gleich, dann immer
145*b1cdbd2cSJim Jagielski 					// beim Ueberlaufen von Start callen
146*b1cdbd2cSJim Jagielski 					if( ( !pEnd || ( nStart == *pEnd ) ) &&
147*b1cdbd2cSJim Jagielski 						( nStart == nCntnt || nStart == nAktCntnt) )
148*b1cdbd2cSJim Jagielski 					{
149*b1cdbd2cSJim Jagielski 						rShell.CallChgLnk();
150*b1cdbd2cSJim Jagielski 						return;
151*b1cdbd2cSJim Jagielski 					}
152*b1cdbd2cSJim Jagielski 
153*b1cdbd2cSJim Jagielski 					// hat das Attribut einen Bereich und dieser nicht leer
154*b1cdbd2cSJim Jagielski 					else if( pEnd && nStart < *pEnd &&
155*b1cdbd2cSJim Jagielski 						// dann teste, ob ueber Start/Ende getravellt wurde
156*b1cdbd2cSJim Jagielski 						( nStart == nCmp ||
157*b1cdbd2cSJim Jagielski 							( pHt->DontExpand() ? nCmp == *pEnd-1
158*b1cdbd2cSJim Jagielski 												: nCmp == *pEnd ) ))
159*b1cdbd2cSJim Jagielski 					{
160*b1cdbd2cSJim Jagielski 						rShell.CallChgLnk();
161*b1cdbd2cSJim Jagielski 						return;
162*b1cdbd2cSJim Jagielski 					}
163*b1cdbd2cSJim Jagielski 					nStart = 0;
164*b1cdbd2cSJim Jagielski 				}
165*b1cdbd2cSJim Jagielski 			}
166*b1cdbd2cSJim Jagielski 
167*b1cdbd2cSJim Jagielski 			if( pBreakIt->GetBreakIter().is() )
168*b1cdbd2cSJim Jagielski 			{
169*b1cdbd2cSJim Jagielski 				const String& rTxt = ((SwTxtNode*)pCNd)->GetTxt();
170*b1cdbd2cSJim Jagielski 				if( !nCmp ||
171*b1cdbd2cSJim Jagielski 					pBreakIt->GetBreakIter()->getScriptType( rTxt, nCmp )
172*b1cdbd2cSJim Jagielski 					 != pBreakIt->GetBreakIter()->getScriptType( rTxt, nCmp - 1 ))
173*b1cdbd2cSJim Jagielski 				{
174*b1cdbd2cSJim Jagielski 					rShell.CallChgLnk();
175*b1cdbd2cSJim Jagielski 					return;
176*b1cdbd2cSJim Jagielski 				}
177*b1cdbd2cSJim Jagielski 			}
178*b1cdbd2cSJim Jagielski 		}
179*b1cdbd2cSJim Jagielski 		else
180*b1cdbd2cSJim Jagielski 			/* wenn mit Home/End/.. mehr als 1 Zeichen getravellt, dann
181*b1cdbd2cSJim Jagielski 			 * immer den ChgLnk rufen, denn es kann hier nicht
182*b1cdbd2cSJim Jagielski 			 * festgestellt werden, was sich geaendert; etwas kann
183*b1cdbd2cSJim Jagielski 			 * veraendert sein.
184*b1cdbd2cSJim Jagielski 			 */
185*b1cdbd2cSJim Jagielski 			rShell.CallChgLnk();
186*b1cdbd2cSJim Jagielski 	}
187*b1cdbd2cSJim Jagielski 
188*b1cdbd2cSJim Jagielski 	const SwFrm* pFrm;
189*b1cdbd2cSJim Jagielski 	const SwFlyFrm *pFlyFrm;
190*b1cdbd2cSJim Jagielski 	if( !rShell.ActionPend() && 0 != ( pFrm = pCNd->getLayoutFrm(rShell.GetLayout(),0,0,sal_False) ) &&
191*b1cdbd2cSJim Jagielski 		0 != ( pFlyFrm = pFrm->FindFlyFrm() ) && !rShell.IsTableMode() )
192*b1cdbd2cSJim Jagielski 	{
193*b1cdbd2cSJim Jagielski 		const SwNodeIndex* pIndex = pFlyFrm->GetFmt()->GetCntnt().GetCntntIdx();
194*b1cdbd2cSJim Jagielski 		ASSERT( pIndex, "Fly ohne Cntnt" );
195*b1cdbd2cSJim Jagielski 
196*b1cdbd2cSJim Jagielski 		if (!pIndex)
197*b1cdbd2cSJim Jagielski 		    return;
198*b1cdbd2cSJim Jagielski 
199*b1cdbd2cSJim Jagielski 		const SwNode& rStNd = pIndex->GetNode();
200*b1cdbd2cSJim Jagielski 
201*b1cdbd2cSJim Jagielski 		if( rStNd.EndOfSectionNode()->StartOfSectionIndex() > nNode ||
202*b1cdbd2cSJim Jagielski 			nNode > rStNd.EndOfSectionIndex() )
203*b1cdbd2cSJim Jagielski 			rShell.GetFlyMacroLnk().Call( (void*)pFlyFrm->GetFmt() );
204*b1cdbd2cSJim Jagielski 	}
205*b1cdbd2cSJim Jagielski }
206*b1cdbd2cSJim Jagielski 
getLayoutFrm(const SwRootFrm * pRoot,SwTxtNode & rNd,xub_StrLen nCntPos,sal_Bool bCalcFrm)207*b1cdbd2cSJim Jagielski long SwCallLink::getLayoutFrm( const SwRootFrm* pRoot, SwTxtNode& rNd, xub_StrLen nCntPos, sal_Bool bCalcFrm )
208*b1cdbd2cSJim Jagielski {
209*b1cdbd2cSJim Jagielski 	SwTxtFrm* pFrm = (SwTxtFrm*)rNd.getLayoutFrm(pRoot,0,0,bCalcFrm), *pNext = pFrm;
210*b1cdbd2cSJim Jagielski 	if ( pFrm && !pFrm->IsHiddenNow() )
211*b1cdbd2cSJim Jagielski 	{
212*b1cdbd2cSJim Jagielski 		if( pFrm->HasFollow() )
213*b1cdbd2cSJim Jagielski 			while( 0 != ( pNext = (SwTxtFrm*)pFrm->GetFollow() ) &&
214*b1cdbd2cSJim Jagielski 					nCntPos >= pNext->GetOfst() )
215*b1cdbd2cSJim Jagielski 				pFrm = pNext;
216*b1cdbd2cSJim Jagielski 
217*b1cdbd2cSJim Jagielski 		return pFrm->Frm().Left();
218*b1cdbd2cSJim Jagielski 	}
219*b1cdbd2cSJim Jagielski 	return 0;
220*b1cdbd2cSJim Jagielski }
221*b1cdbd2cSJim Jagielski 
222