nss: upgrade to release 3.73
[LibreOffice.git] / sw / source / core / fields / ddetbl.cxx
blob7954e18f5d4c9fe786b2c05faa039e83fa7d3a96
1 /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
2 /*
3 * This file is part of the LibreOffice project.
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/.
9 * This file incorporates work covered by the following license notice:
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 .
20 #include <doc.hxx>
21 #include <IDocumentSettingAccess.hxx>
22 #include <IDocumentFieldsAccess.hxx>
23 #include <IDocumentLayoutAccess.hxx>
24 #include <index.hxx>
25 #include <ndtxt.hxx>
26 #include <swtable.hxx>
27 #include <swddetbl.hxx>
28 #include <fmtfld.hxx>
29 #include <ddefld.hxx>
30 #include <ndindex.hxx>
31 #include <fldupde.hxx>
32 #include <swtblfmt.hxx>
33 #include <fieldhint.hxx>
34 #include <osl/diagnose.h>
35 #include <pam.hxx>
37 /// Ctor moves all lines/boxes from a SwTable into itself.
38 /// Afterwards the SwTable is empty and must be deleted.
39 SwDDETable::SwDDETable( SwTable& rTable, SwDDEFieldType* pDDEType, bool bUpdate )
40 : SwTable(rTable), m_aDepends(*this), m_pDDEType(pDDEType)
42 m_aDepends.StartListening(m_pDDEType);
43 // copy the table data
44 m_TabSortContentBoxes.insert(rTable.GetTabSortBoxes());
45 rTable.GetTabSortBoxes().clear();
47 m_aLines.insert( m_aLines.begin(),
48 rTable.GetTabLines().begin(), rTable.GetTabLines().end() ); // move lines
49 rTable.GetTabLines().clear();
51 if( !m_aLines.empty() )
53 const SwNode& rNd = *GetTabSortBoxes()[0]->GetSttNd();
54 if( rNd.GetNodes().IsDocNodes() )
56 pDDEType->IncRefCnt();
58 // update box content only if update flag is set (false in import)
59 if (bUpdate)
60 ChangeContent();
65 SwDDETable::~SwDDETable()
67 SwDoc* pDoc = GetFrameFormat()->GetDoc();
68 if (!pDoc->IsInDtor() && !m_aLines.empty())
70 assert(m_pTableNode);
71 if (m_pTableNode->GetNodes().IsDocNodes())
73 m_pDDEType->DecRefCnt();
77 // If it is the last dependent of the "deleted field" than delete it finally
78 if( m_pDDEType->IsDeleted() && m_pDDEType->HasOnlyOneListener() )
80 m_aDepends.EndListeningAll();
81 delete m_pDDEType;
82 m_pDDEType = nullptr;
86 void SwDDETable::Modify( const SfxPoolItem* pOld, const SfxPoolItem* pNew )
88 if( pNew && RES_UPDATEDDETBL == pNew->Which() )
89 ChangeContent();
90 else
91 SwTable::Modify( pOld, pNew );
94 void SwDDETable::SwClientNotify( const SwModify& rModify, const SfxHint& rHint )
96 SwClient::SwClientNotify(rModify, rHint);
97 if(auto pFieldHint = dynamic_cast<const SwFieldHint*>(&rHint))
99 pFieldHint->m_pPaM->DeleteMark(); // TODO: this is really hackish
100 // replace DDETable by real table
101 NoDDETable();
103 else if(const auto pLinkAnchorHint = dynamic_cast<const sw::LinkAnchorSearchHint*>(&rHint))
105 if(pLinkAnchorHint->m_rpFoundNode)
106 return;
107 const auto pNd = GetTabSortBoxes()[0]->GetSttNd();
108 if( pNd && &pLinkAnchorHint->m_rNodes == &pNd->GetNodes() )
109 pLinkAnchorHint->m_rpFoundNode = pNd;
111 else if(const sw::InRangeSearchHint* pInRangeHint = dynamic_cast<const sw::InRangeSearchHint*>(&rHint))
113 if(pInRangeHint->m_rIsInRange)
114 return;
115 const SwTableNode* pTableNd = GetTabSortBoxes()[0]->GetSttNd()->FindTableNode();
116 if( pTableNd->GetNodes().IsDocNodes() &&
117 pInRangeHint->m_nSttNd < pTableNd->EndOfSectionIndex() &&
118 pInRangeHint->m_nEndNd > pTableNd->GetIndex() )
119 pInRangeHint->m_rIsInRange = true;
120 } else if (const auto pGatherDdeTablesHint = dynamic_cast<const sw::GatherDdeTablesHint*>(&rHint))
122 pGatherDdeTablesHint->m_rvTables.push_back(this);
124 else if (auto pModifyChangedHint = dynamic_cast<const sw::ModifyChangedHint*>(&rHint))
126 if(m_pDDEType == &rModify)
127 m_pDDEType = const_cast<SwDDEFieldType*>(static_cast<const SwDDEFieldType*>(pModifyChangedHint->m_pNew));
131 void SwDDETable::ChangeContent()
133 OSL_ENSURE( GetFrameFormat(), "No FrameFormat" );
135 // Is this the correct NodesArray? (because of UNDO)
136 if( m_aLines.empty() )
137 return;
138 OSL_ENSURE( !GetTabSortBoxes().empty(), "Table without content?" );
139 if( !GetTabSortBoxes()[0]->GetSttNd()->GetNodes().IsDocNodes() )
140 return;
143 OUString aExpand = m_pDDEType->GetExpansion().replaceAll("\r", "");
144 sal_Int32 nExpandTokenPos = 0;
146 for( size_t n = 0; n < m_aLines.size(); ++n )
148 OUString aLine = aExpand.getToken( 0, '\n', nExpandTokenPos );
149 sal_Int32 nLineTokenPos = 0;
150 SwTableLine* pLine = m_aLines[ n ];
151 for( size_t i = 0; i < pLine->GetTabBoxes().size(); ++i )
153 SwTableBox* pBox = pLine->GetTabBoxes()[ i ];
154 OSL_ENSURE( pBox->GetSttIdx(), "no content box" );
155 SwNodeIndex aNdIdx( *pBox->GetSttNd(), 1 );
156 SwTextNode* pTextNode = aNdIdx.GetNode().GetTextNode();
157 OSL_ENSURE( pTextNode, "No Node" );
158 SwIndex aCntIdx( pTextNode, 0 );
159 pTextNode->EraseText( aCntIdx );
160 pTextNode->InsertText( aLine.getToken( 0, '\t', nLineTokenPos ), aCntIdx );
162 SwTableBoxFormat* pBoxFormat = static_cast<SwTableBoxFormat*>(pBox->GetFrameFormat());
163 pBoxFormat->LockModify();
164 pBoxFormat->ResetFormatAttr( RES_BOXATR_VALUE );
165 pBoxFormat->UnlockModify();
169 const IDocumentSettingAccess& rIDSA = GetFrameFormat()->getIDocumentSettingAccess();
170 SwDoc* pDoc = GetFrameFormat()->GetDoc();
171 if( AUTOUPD_FIELD_AND_CHARTS == rIDSA.getFieldUpdateFlags(true) )
172 pDoc->getIDocumentFieldsAccess().SetFieldsDirty( true, nullptr, 0 );
175 SwDDEFieldType* SwDDETable::GetDDEFieldType()
177 return m_pDDEType;
180 bool SwDDETable::NoDDETable()
182 // search table node
183 OSL_ENSURE( GetFrameFormat(), "No FrameFormat" );
184 SwDoc* pDoc = GetFrameFormat()->GetDoc();
186 // Is this the correct NodesArray? (because of UNDO)
187 if( m_aLines.empty() )
188 return false;
189 OSL_ENSURE( !GetTabSortBoxes().empty(), "Table without content?" );
190 SwNode* pNd = const_cast<SwNode*>(static_cast<SwNode const *>(GetTabSortBoxes()[0]->GetSttNd()));
191 if( !pNd->GetNodes().IsDocNodes() )
192 return false;
194 SwTableNode* pTableNd = pNd->FindTableNode();
195 OSL_ENSURE( pTableNd, "Where is the table?");
197 std::unique_ptr<SwTable> pNewTable(new SwTable( *this ));
199 // copy the table data
200 pNewTable->GetTabSortBoxes().insert( GetTabSortBoxes() ); // move content boxes
201 GetTabSortBoxes().clear();
203 pNewTable->GetTabLines().insert( pNewTable->GetTabLines().begin(),
204 GetTabLines().begin(), GetTabLines().end() ); // move lines
205 GetTabLines().clear();
207 if( pDoc->getIDocumentLayoutAccess().GetCurrentViewShell() )
208 m_pDDEType->DecRefCnt();
210 pTableNd->SetNewTable( std::move(pNewTable) ); // replace table
212 return true;
215 /* vim:set shiftwidth=4 softtabstop=4 expandtab: */