1 /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
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 .
21 #include <IDocumentSettingAccess.hxx>
22 #include <IDocumentFieldsAccess.hxx>
23 #include <IDocumentLayoutAccess.hxx>
26 #include <swtable.hxx>
27 #include <swddetbl.hxx>
30 #include <ndindex.hxx>
31 #include <fldupde.hxx>
32 #include <swtblfmt.hxx>
33 #include <fieldhint.hxx>
34 #include <osl/diagnose.h>
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)
65 SwDDETable::~SwDDETable()
67 SwDoc
* pDoc
= GetFrameFormat()->GetDoc();
68 if (!pDoc
->IsInDtor() && !m_aLines
.empty())
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();
86 void SwDDETable::Modify( const SfxPoolItem
* pOld
, const SfxPoolItem
* pNew
)
88 if( pNew
&& RES_UPDATEDDETBL
== pNew
->Which() )
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
103 else if(const auto pLinkAnchorHint
= dynamic_cast<const sw::LinkAnchorSearchHint
*>(&rHint
))
105 if(pLinkAnchorHint
->m_rpFoundNode
)
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
)
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() )
138 OSL_ENSURE( !GetTabSortBoxes().empty(), "Table without content?" );
139 if( !GetTabSortBoxes()[0]->GetSttNd()->GetNodes().IsDocNodes() )
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()
180 bool SwDDETable::NoDDETable()
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() )
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() )
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
215 /* vim:set shiftwidth=4 softtabstop=4 expandtab: */