Line data Source code
1 : /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
2 : /*
3 : * This file is part of the LibreOffice project.
4 : *
5 : * This Source Code Form is subject to the terms of the Mozilla Public
6 : * License, v. 2.0. If a copy of the MPL was not distributed with this
7 : * file, You can obtain one at http://mozilla.org/MPL/2.0/.
8 : *
9 : * This file incorporates work covered by the following license notice:
10 : *
11 : * Licensed to the Apache Software Foundation (ASF) under one or more
12 : * contributor license agreements. See the NOTICE file distributed
13 : * with this work for additional information regarding copyright
14 : * ownership. The ASF licenses this file to you under the Apache
15 : * License, Version 2.0 (the "License"); you may not use this file
16 : * except in compliance with the License. You may obtain a copy of
17 : * the License at http://www.apache.org/licenses/LICENSE-2.0 .
18 : */
19 :
20 : #include <doc.hxx>
21 : #include <IDocumentUndoRedo.hxx>
22 : #include <IDocumentSettingAccess.hxx>
23 : #include <IDocumentState.hxx>
24 : #include <editsh.hxx>
25 : #include <pam.hxx>
26 : #include <ndtxt.hxx>
27 : #include <docary.hxx>
28 : #include <swwait.hxx>
29 : #include <swundo.hxx>
30 : #include <section.hxx>
31 : #include <doctxm.hxx>
32 : #include <edglbldc.hxx>
33 :
34 25 : bool SwEditShell::IsGlobalDoc() const
35 : {
36 25 : return getIDocumentSettingAccess()->get(DocumentSettingId::GLOBAL_DOCUMENT);
37 : }
38 :
39 0 : void SwEditShell::SetGlblDocSaveLinks( bool bFlag )
40 : {
41 0 : getIDocumentSettingAccess()->set(DocumentSettingId::GLOBAL_DOCUMENT_SAVE_LINKS, bFlag);
42 0 : if( !GetDoc()->getIDocumentState().IsModified() ) // Bug 57028
43 : {
44 0 : GetDoc()->GetIDocumentUndoRedo().SetUndoNoResetModified();
45 : }
46 0 : GetDoc()->getIDocumentState().SetModified();
47 0 : }
48 :
49 0 : bool SwEditShell::IsGlblDocSaveLinks() const
50 : {
51 0 : return getIDocumentSettingAccess()->get(DocumentSettingId::GLOBAL_DOCUMENT_SAVE_LINKS);
52 : }
53 :
54 0 : void SwEditShell::GetGlobalDocContent( SwGlblDocContents& rArr ) const
55 : {
56 0 : rArr.DeleteAndDestroyAll();
57 :
58 0 : if( !getIDocumentSettingAccess()->get(DocumentSettingId::GLOBAL_DOCUMENT) )
59 0 : return;
60 :
61 : // then all linked areas on the topmost level
62 0 : SwDoc* pMyDoc = GetDoc();
63 0 : const SwSectionFormats& rSectFormats = pMyDoc->GetSections();
64 :
65 0 : for( auto n = rSectFormats.size(); n; )
66 : {
67 0 : const SwSection* pSect = rSectFormats[ --n ]->GetGlobalDocSection();
68 0 : if( pSect )
69 : {
70 : SwGlblDocContent* pNew;
71 0 : switch( pSect->GetType() )
72 : {
73 : case TOX_HEADER_SECTION:
74 0 : break; // ignore
75 : case TOX_CONTENT_SECTION:
76 : OSL_ENSURE( pSect->ISA( SwTOXBaseSection ), "no TOXBaseSection!" );
77 0 : pNew = new SwGlblDocContent( static_cast<const SwTOXBaseSection*>(pSect) );
78 0 : break;
79 :
80 : default:
81 0 : pNew = new SwGlblDocContent( pSect );
82 0 : break;
83 : }
84 0 : if( !rArr.insert( pNew ).second )
85 0 : delete pNew;
86 : }
87 : }
88 :
89 : // and finally add the dummies (other text)
90 : SwNode* pNd;
91 0 : sal_uLong nSttIdx = pMyDoc->GetNodes().GetEndOfExtras().GetIndex() + 2;
92 0 : for( SwGlblDocContents::size_type n = 0; n < rArr.size(); ++n )
93 : {
94 0 : const SwGlblDocContent& rNew = *rArr[ n ];
95 : // Search from StartPos until rNew.DocPos for a content node.
96 : // If one exists then a dummy entry is needed.
97 0 : for( ; nSttIdx < rNew.GetDocPos(); ++nSttIdx )
98 0 : if( ( pNd = pMyDoc->GetNodes()[ nSttIdx ])->IsContentNode()
99 0 : || pNd->IsSectionNode() || pNd->IsTableNode() )
100 : {
101 0 : SwGlblDocContent* pNew = new SwGlblDocContent( nSttIdx );
102 0 : if( !rArr.insert( pNew ).second )
103 0 : delete pNew;
104 : else
105 0 : ++n; // to the next position
106 0 : break;
107 : }
108 :
109 : // set StartPosition to the end
110 0 : nSttIdx = pMyDoc->GetNodes()[ rNew.GetDocPos() ]->EndOfSectionIndex();
111 0 : ++nSttIdx;
112 : }
113 :
114 : // Should the end also be set?
115 0 : if( !rArr.empty() )
116 : {
117 0 : sal_uLong nNdEnd = pMyDoc->GetNodes().GetEndOfContent().GetIndex();
118 0 : for( ; nSttIdx < nNdEnd; ++nSttIdx )
119 0 : if( ( pNd = pMyDoc->GetNodes()[ nSttIdx ])->IsContentNode()
120 0 : || pNd->IsSectionNode() || pNd->IsTableNode() )
121 : {
122 0 : SwGlblDocContent* pNew = new SwGlblDocContent( nSttIdx );
123 0 : if( !rArr.insert( pNew ).second )
124 0 : delete pNew;
125 0 : break;
126 : }
127 : }
128 : else
129 : {
130 : SwGlblDocContent* pNew = new SwGlblDocContent(
131 0 : pMyDoc->GetNodes().GetEndOfExtras().GetIndex() + 2 );
132 0 : rArr.insert( pNew );
133 : }
134 : }
135 :
136 0 : bool SwEditShell::InsertGlobalDocContent( const SwGlblDocContent& rInsPos,
137 : SwSectionData & rNew)
138 : {
139 0 : if( !getIDocumentSettingAccess()->get(DocumentSettingId::GLOBAL_DOCUMENT) )
140 0 : return false;
141 :
142 0 : SET_CURR_SHELL( this );
143 0 : StartAllAction();
144 :
145 0 : SwPaM* pCrsr = GetCrsr();
146 0 : if( pCrsr->GetNext() != pCrsr || IsTableMode() )
147 0 : ClearMark();
148 :
149 0 : SwPosition& rPos = *pCrsr->GetPoint();
150 0 : rPos.nNode = rInsPos.GetDocPos();
151 :
152 0 : bool bEndUndo = false;
153 0 : SwDoc* pMyDoc = GetDoc();
154 0 : SwTextNode *const pTextNd = rPos.nNode.GetNode().GetTextNode();
155 0 : if( pTextNd )
156 0 : rPos.nContent.Assign( pTextNd, 0 );
157 : else
158 : {
159 0 : bEndUndo = true;
160 0 : pMyDoc->GetIDocumentUndoRedo().StartUndo( UNDO_START, NULL );
161 0 : --rPos.nNode;
162 0 : pMyDoc->getIDocumentContentOperations().AppendTextNode( rPos );
163 0 : pCrsr->SetMark();
164 : }
165 :
166 0 : InsertSection( rNew );
167 :
168 0 : if( bEndUndo )
169 : {
170 0 : pMyDoc->GetIDocumentUndoRedo().EndUndo( UNDO_END, NULL );
171 : }
172 0 : EndAllAction();
173 :
174 0 : return true;
175 : }
176 :
177 0 : bool SwEditShell::InsertGlobalDocContent( const SwGlblDocContent& rInsPos,
178 : const SwTOXBase& rTOX )
179 : {
180 0 : if( !getIDocumentSettingAccess()->get(DocumentSettingId::GLOBAL_DOCUMENT) )
181 0 : return false;
182 :
183 0 : SET_CURR_SHELL( this );
184 0 : StartAllAction();
185 :
186 0 : SwPaM* pCrsr = GetCrsr();
187 0 : if( pCrsr->GetNext() != pCrsr || IsTableMode() )
188 0 : ClearMark();
189 :
190 0 : SwPosition& rPos = *pCrsr->GetPoint();
191 0 : rPos.nNode = rInsPos.GetDocPos();
192 :
193 0 : bool bEndUndo = false;
194 0 : SwDoc* pMyDoc = GetDoc();
195 0 : SwTextNode* pTextNd = rPos.nNode.GetNode().GetTextNode();
196 0 : if (pTextNd && pTextNd->GetText().getLength() && rPos.nNode.GetIndex() + 1 !=
197 0 : pMyDoc->GetNodes().GetEndOfContent().GetIndex() )
198 0 : rPos.nContent.Assign( pTextNd, 0 );
199 : else
200 : {
201 0 : bEndUndo = true;
202 0 : pMyDoc->GetIDocumentUndoRedo().StartUndo( UNDO_START, NULL );
203 0 : --rPos.nNode;
204 0 : pMyDoc->getIDocumentContentOperations().AppendTextNode( rPos );
205 : }
206 :
207 0 : InsertTableOf( rTOX );
208 :
209 0 : if( bEndUndo )
210 : {
211 0 : pMyDoc->GetIDocumentUndoRedo().EndUndo( UNDO_END, NULL );
212 : }
213 0 : EndAllAction();
214 :
215 0 : return true;
216 : }
217 :
218 0 : bool SwEditShell::InsertGlobalDocContent( const SwGlblDocContent& rInsPos )
219 : {
220 0 : if( !getIDocumentSettingAccess()->get(DocumentSettingId::GLOBAL_DOCUMENT) )
221 0 : return false;
222 :
223 0 : SET_CURR_SHELL( this );
224 0 : StartAllAction();
225 :
226 0 : SwPaM* pCrsr = GetCrsr();
227 0 : if( pCrsr->GetNext() != pCrsr || IsTableMode() )
228 0 : ClearMark();
229 :
230 0 : SwPosition& rPos = *pCrsr->GetPoint();
231 0 : rPos.nNode = rInsPos.GetDocPos() - 1;
232 0 : rPos.nContent.Assign( 0, 0 );
233 :
234 0 : SwDoc* pMyDoc = GetDoc();
235 0 : pMyDoc->getIDocumentContentOperations().AppendTextNode( rPos );
236 0 : EndAllAction();
237 0 : return true;
238 : }
239 :
240 0 : bool SwEditShell::DeleteGlobalDocContent( const SwGlblDocContents& rArr ,
241 : size_t nDelPos )
242 : {
243 0 : if( !getIDocumentSettingAccess()->get(DocumentSettingId::GLOBAL_DOCUMENT) )
244 0 : return false;
245 :
246 0 : SET_CURR_SHELL( this );
247 0 : StartAllAction();
248 0 : StartUndo( UNDO_START );
249 :
250 0 : SwPaM* pCrsr = GetCrsr();
251 0 : if( pCrsr->GetNext() != pCrsr || IsTableMode() )
252 0 : ClearMark();
253 :
254 0 : SwPosition& rPos = *pCrsr->GetPoint();
255 :
256 0 : SwDoc* pMyDoc = GetDoc();
257 0 : const SwGlblDocContent& rDelPos = *rArr[ nDelPos ];
258 0 : sal_uLong nDelIdx = rDelPos.GetDocPos();
259 0 : if( 1 == rArr.size() )
260 : {
261 : // we need at least one node!
262 0 : rPos.nNode = nDelIdx - 1;
263 0 : rPos.nContent.Assign( 0, 0 );
264 :
265 0 : pMyDoc->getIDocumentContentOperations().AppendTextNode( rPos );
266 0 : ++nDelIdx;
267 : }
268 :
269 0 : switch( rDelPos.GetType() )
270 : {
271 : case GLBLDOC_UNKNOWN:
272 : {
273 0 : rPos.nNode = nDelIdx;
274 0 : pCrsr->SetMark();
275 0 : if( ++nDelPos < rArr.size() )
276 0 : rPos.nNode = rArr[ nDelPos ]->GetDocPos();
277 : else
278 0 : rPos.nNode = pMyDoc->GetNodes().GetEndOfContent();
279 0 : --rPos.nNode;
280 0 : if( !pMyDoc->getIDocumentContentOperations().DelFullPara( *pCrsr ) )
281 0 : Delete();
282 : }
283 0 : break;
284 :
285 : case GLBLDOC_TOXBASE:
286 : {
287 0 : const SwTOXBaseSection* pTOX = static_cast<const SwTOXBaseSection*>(rDelPos.GetTOX());
288 0 : pMyDoc->DeleteTOX( *pTOX, true );
289 : }
290 0 : break;
291 :
292 : case GLBLDOC_SECTION:
293 : {
294 0 : SwSectionFormat* pSectFormat = const_cast<SwSectionFormat*>(rDelPos.GetSection()->GetFormat());
295 0 : pMyDoc->DelSectionFormat( pSectFormat, true );
296 : }
297 0 : break;
298 : }
299 :
300 0 : EndUndo( UNDO_END );
301 0 : EndAllAction();
302 0 : return true;
303 : }
304 :
305 0 : bool SwEditShell::MoveGlobalDocContent( const SwGlblDocContents& rArr ,
306 : size_t nFromPos, size_t nToPos,
307 : size_t nInsPos )
308 : {
309 0 : if( !getIDocumentSettingAccess()->get(DocumentSettingId::GLOBAL_DOCUMENT) ||
310 0 : nFromPos >= rArr.size() || nToPos > rArr.size() ||
311 0 : nInsPos > rArr.size() || nFromPos >= nToPos ||
312 0 : ( nFromPos <= nInsPos && nInsPos <= nToPos ) )
313 0 : return false;
314 :
315 0 : SET_CURR_SHELL( this );
316 0 : StartAllAction();
317 :
318 0 : SwPaM* pCrsr = GetCrsr();
319 0 : if( pCrsr->GetNext() != pCrsr || IsTableMode() )
320 0 : ClearMark();
321 :
322 0 : SwDoc* pMyDoc = GetDoc();
323 0 : SwNodeRange aRg( pMyDoc->GetNodes(), rArr[ nFromPos ]->GetDocPos() );
324 0 : if( nToPos < rArr.size() )
325 0 : aRg.aEnd = rArr[ nToPos ]->GetDocPos();
326 : else
327 0 : aRg.aEnd = pMyDoc->GetNodes().GetEndOfContent();
328 :
329 0 : SwNodeIndex aInsPos( pMyDoc->GetNodes() );
330 0 : if( nInsPos < rArr.size() )
331 0 : aInsPos = rArr[ nInsPos ]->GetDocPos();
332 : else
333 0 : aInsPos = pMyDoc->GetNodes().GetEndOfContent();
334 :
335 0 : bool bRet = pMyDoc->getIDocumentContentOperations().MoveNodeRange( aRg, aInsPos,
336 0 : SwMoveFlags::ALLFLYS | SwMoveFlags::CREATEUNDOOBJ );
337 :
338 0 : EndAllAction();
339 0 : return bRet;
340 : }
341 :
342 0 : bool SwEditShell::GotoGlobalDocContent( const SwGlblDocContent& rPos )
343 : {
344 0 : if( !getIDocumentSettingAccess()->get(DocumentSettingId::GLOBAL_DOCUMENT) )
345 0 : return false;
346 :
347 0 : SET_CURR_SHELL( this );
348 0 : SttCrsrMove();
349 :
350 0 : SwPaM* pCrsr = GetCrsr();
351 0 : if( pCrsr->GetNext() != pCrsr || IsTableMode() )
352 0 : ClearMark();
353 :
354 0 : SwPosition& rCrsrPos = *pCrsr->GetPoint();
355 0 : rCrsrPos.nNode = rPos.GetDocPos();
356 :
357 0 : SwDoc* pMyDoc = GetDoc();
358 0 : SwContentNode * pCNd = rCrsrPos.nNode.GetNode().GetContentNode();
359 0 : if( !pCNd )
360 0 : pCNd = pMyDoc->GetNodes().GoNext( &rCrsrPos.nNode );
361 :
362 0 : rCrsrPos.nContent.Assign( pCNd, 0 );
363 :
364 0 : EndCrsrMove();
365 0 : return true;
366 : }
367 :
368 0 : SwGlblDocContent::SwGlblDocContent( sal_uLong nPos )
369 : {
370 0 : eType = GLBLDOC_UNKNOWN;
371 0 : PTR.pTOX = 0;
372 0 : nDocPos = nPos;
373 0 : }
374 :
375 0 : SwGlblDocContent::SwGlblDocContent( const SwTOXBaseSection* pTOX )
376 : {
377 0 : eType = GLBLDOC_TOXBASE;
378 0 : PTR.pTOX = pTOX;
379 :
380 0 : const SwSectionNode* pSectNd = pTOX->GetFormat()->GetSectionNode();
381 0 : nDocPos = pSectNd ? pSectNd->GetIndex() : 0;
382 0 : }
383 :
384 0 : SwGlblDocContent::SwGlblDocContent( const SwSection* pSect )
385 : {
386 0 : eType = GLBLDOC_SECTION;
387 0 : PTR.pSect = pSect;
388 :
389 0 : const SwSectionNode* pSectNd = pSect->GetFormat()->GetSectionNode();
390 0 : nDocPos = pSectNd ? pSectNd->GetIndex() : 0;
391 177 : }
392 :
393 : /* vim:set shiftwidth=4 softtabstop=4 expandtab: */
|