Version 7.6.3.2-android, tag libreoffice-7.6.3.2-android
[LibreOffice.git] / sw / source / core / fields / expfld.cxx
blob39467d9b7a61853d5f8e1fa98d8e6191eb239f2f
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 <sal/config.h>
22 #include <limits>
24 #include <UndoTable.hxx>
25 #include <hintids.hxx>
26 #include <o3tl/any.hxx>
27 #include <osl/diagnose.h>
28 #include <unotools/collatorwrapper.hxx>
29 #include <unotools/charclass.hxx>
30 #include <editeng/langitem.hxx>
31 #include <editeng/fontitem.hxx>
32 #include <com/sun/star/text/SetVariableType.hpp>
33 #include <unofield.hxx>
34 #include <frmfmt.hxx>
35 #include <fmtfld.hxx>
36 #include <txtfld.hxx>
37 #include <fmtanchr.hxx>
38 #include <txtftn.hxx>
39 #include <doc.hxx>
40 #include <IDocumentFieldsAccess.hxx>
41 #include <layfrm.hxx>
42 #include <pagefrm.hxx>
43 #include <cntfrm.hxx>
44 #include <txtfrm.hxx>
45 #include <rootfrm.hxx>
46 #include <tabfrm.hxx>
47 #include <flyfrm.hxx>
48 #include <ftnfrm.hxx>
49 #include <rowfrm.hxx>
50 #include <expfld.hxx>
51 #include <usrfld.hxx>
52 #include <ndtxt.hxx>
53 #include <calc.hxx>
54 #include <pam.hxx>
55 #include <docfld.hxx>
56 #include <swtable.hxx>
57 #include <breakit.hxx>
58 #include <SwStyleNameMapper.hxx>
59 #include <unofldmid.h>
60 #include <numrule.hxx>
61 #include <utility>
63 using namespace ::com::sun::star;
64 using namespace ::com::sun::star::text;
66 static sal_Int16 lcl_SubTypeToAPI(sal_uInt16 nSubType)
68 sal_Int16 nRet = 0;
69 switch(nSubType)
71 case nsSwGetSetExpType::GSE_EXPR:
72 nRet = SetVariableType::VAR; // 0
73 break;
74 case nsSwGetSetExpType::GSE_SEQ:
75 nRet = SetVariableType::SEQUENCE; // 1
76 break;
77 case nsSwGetSetExpType::GSE_FORMULA:
78 nRet = SetVariableType::FORMULA; // 2
79 break;
80 case nsSwGetSetExpType::GSE_STRING:
81 nRet = SetVariableType::STRING; // 3
82 break;
84 return nRet;
87 static sal_Int32 lcl_APIToSubType(const uno::Any& rAny)
89 sal_Int16 nVal = 0;
90 rAny >>= nVal;
91 sal_Int32 nSet = 0;
92 switch(nVal)
94 case SetVariableType::VAR: nSet = nsSwGetSetExpType::GSE_EXPR; break;
95 case SetVariableType::SEQUENCE: nSet = nsSwGetSetExpType::GSE_SEQ; break;
96 case SetVariableType::FORMULA: nSet = nsSwGetSetExpType::GSE_FORMULA; break;
97 case SetVariableType::STRING: nSet = nsSwGetSetExpType::GSE_STRING; break;
98 default:
99 OSL_FAIL("wrong value");
100 nSet = -1;
102 return nSet;
105 OUString ReplacePoint( const OUString& rTmpName, bool bWithCommandType )
107 // replace first and last (if bWithCommandType: last two) dot
108 // since table names may contain dots
110 sal_Int32 nIndex = rTmpName.lastIndexOf('.');
111 if (nIndex<0)
113 return rTmpName;
116 OUString sRes = rTmpName.replaceAt(nIndex, 1, rtl::OUStringChar(DB_DELIM));
118 if (bWithCommandType)
120 nIndex = sRes.lastIndexOf('.', nIndex);
121 if (nIndex<0)
123 return sRes;
125 sRes = sRes.replaceAt(nIndex, 1, rtl::OUStringChar(DB_DELIM));
128 nIndex = sRes.indexOf('.');
129 if (nIndex>=0)
131 sRes = sRes.replaceAt(nIndex, 1, rtl::OUStringChar(DB_DELIM));
133 return sRes;
136 static SwTextNode* GetFirstTextNode( const SwDoc& rDoc, SwPosition& rPos,
137 const SwContentFrame *pCFrame, Point &rPt )
139 SwTextNode* pTextNode = nullptr;
140 if ( !pCFrame )
142 const SwNodes& rNodes = rDoc.GetNodes();
143 rPos.Assign( *rNodes.GetEndOfContent().StartOfSectionNode() );
144 SwContentNode* pCNd;
145 while( nullptr != (pCNd = rNodes.GoNext( &rPos ) ) &&
146 nullptr == ( pTextNode = pCNd->GetTextNode() ) )
148 OSL_ENSURE( pTextNode, "Where is the 1. TextNode?" );
150 else if ( !pCFrame->isFrameAreaDefinitionValid() )
152 assert(pCFrame->IsTextFrame());
153 rPos = static_cast<SwTextFrame const*>(pCFrame)->MapViewToModelPos(TextFrameIndex(0));
155 else
157 pCFrame->GetModelPositionForViewPoint( &rPos, rPt );
158 pTextNode = rPos.GetNode().GetTextNode();
160 return pTextNode;
163 const SwTextNode* GetBodyTextNode( const SwDoc& rDoc, SwPosition& rPos,
164 const SwFrame& rFrame )
166 const SwLayoutFrame* pLayout = rFrame.GetUpper();
167 const SwTextNode* pTextNode = nullptr;
169 while( pLayout )
171 if( pLayout->IsFlyFrame() )
173 // get the FlyFormat
174 const SwFrameFormat* pFlyFormat = static_cast<const SwFlyFrame*>(pLayout)->GetFormat();
175 OSL_ENSURE( pFlyFormat, "Could not find FlyFormat, where is the field?" );
177 const SwFormatAnchor &rAnchor = pFlyFormat->GetAnchor();
179 if( RndStdIds::FLY_AT_FLY == rAnchor.GetAnchorId() )
181 // the fly needs to be attached somewhere, so ask it
182 pLayout = static_cast<const SwLayoutFrame*>(static_cast<const SwFlyFrame*>(pLayout)->GetAnchorFrame());
183 continue;
185 else if ((RndStdIds::FLY_AT_PARA == rAnchor.GetAnchorId()) ||
186 (RndStdIds::FLY_AT_CHAR == rAnchor.GetAnchorId()) ||
187 (RndStdIds::FLY_AS_CHAR == rAnchor.GetAnchorId()))
189 OSL_ENSURE( rAnchor.GetContentAnchor(), "no valid position" );
190 rPos = *rAnchor.GetContentAnchor();
191 pTextNode = rPos.GetNode().GetTextNode();
192 if ( RndStdIds::FLY_AT_PARA == rAnchor.GetAnchorId() )
194 rPos.AssignStartIndex(*pTextNode);
197 // do not break yet, might be as well in Header/Footer/Footnote/Fly
198 pLayout = static_cast<const SwFlyFrame*>(pLayout)->GetAnchorFrame()
199 ? static_cast<const SwFlyFrame*>(pLayout)->GetAnchorFrame()->GetUpper() : nullptr;
200 continue;
202 else
204 pLayout->FindPageFrame()->GetContentPosition(
205 pLayout->getFrameArea().Pos(), rPos );
206 pTextNode = rPos.GetNode().GetTextNode();
209 else if( pLayout->IsFootnoteFrame() )
211 // get the anchor's node
212 const SwTextFootnote* pFootnote = static_cast<const SwFootnoteFrame*>(pLayout)->GetAttr();
213 pTextNode = &pFootnote->GetTextNode();
214 rPos.Assign( *pTextNode, pFootnote->GetStart() );
216 else if( pLayout->IsHeaderFrame() || pLayout->IsFooterFrame() )
218 const SwContentFrame* pContentFrame;
219 const SwPageFrame* pPgFrame = pLayout->FindPageFrame();
220 if( pLayout->IsHeaderFrame() )
222 const SwTabFrame *pTab;
223 if( nullptr != ( pContentFrame = pPgFrame->FindFirstBodyContent()) &&
224 nullptr != (pTab = pContentFrame->FindTabFrame()) && pTab->IsFollow() &&
225 pTab->GetTable()->GetRowsToRepeat() > 0 &&
226 pTab->IsInHeadline( *pContentFrame ) )
228 // take the next line
229 const SwLayoutFrame* pRow = pTab->GetFirstNonHeadlineRow();
230 pContentFrame = pRow->ContainsContent();
233 else
234 pContentFrame = pPgFrame->FindLastBodyContent();
236 if( pContentFrame )
238 assert(pContentFrame->IsTextFrame());
239 SwTextFrame const*const pFrame(static_cast<SwTextFrame const*>(pContentFrame));
240 rPos = pFrame->MapViewToModelPos(TextFrameIndex(pFrame->GetText().getLength()));
241 pTextNode = rPos.GetNode().GetTextNode();
242 assert(pTextNode);
244 else
246 Point aPt( pLayout->getFrameArea().Pos() );
247 aPt.AdjustY( 1 ); // get out of the header
248 pContentFrame = pPgFrame->GetContentPos( aPt, false, true );
249 pTextNode = GetFirstTextNode( rDoc, rPos, pContentFrame, aPt );
252 else
254 pLayout = pLayout->GetUpper();
255 continue;
257 break; // found, so finish loop
259 return pTextNode;
262 SwGetExpFieldType::SwGetExpFieldType(SwDoc* pDc)
263 : SwValueFieldType( pDc, SwFieldIds::GetExp )
267 std::unique_ptr<SwFieldType> SwGetExpFieldType::Copy() const
269 return std::make_unique<SwGetExpFieldType>(GetDoc());
272 void SwGetExpFieldType::SwClientNotify(const SwModify&, const SfxHint&)
274 // do not expand anything (else)
277 SwGetExpField::SwGetExpField(SwGetExpFieldType* pTyp, const OUString& rFormel,
278 sal_uInt16 nSub, sal_uLong nFormat)
279 : SwFormulaField( pTyp, nFormat, 0.0 )
280 , m_fValueRLHidden(0.0)
282 m_bIsInBodyText( true ),
283 m_nSubType(nSub),
284 m_bLateInitialization( false )
286 SetFormula( rFormel );
289 void SwGetExpField::ChgExpStr(const OUString& rExpand, SwRootFrame const*const pLayout)
291 if (!pLayout || pLayout->IsHideRedlines())
293 m_sExpandRLHidden = rExpand;
295 if (!pLayout || !pLayout->IsHideRedlines())
297 m_sExpand = rExpand;
301 OUString SwGetExpField::ExpandImpl(SwRootFrame const*const pLayout) const
303 if(m_nSubType & nsSwExtendedSubType::SUB_CMD)
304 return GetFormula();
306 return (pLayout && pLayout->IsHideRedlines()) ? m_sExpandRLHidden : m_sExpand;
309 OUString SwGetExpField::GetFieldName() const
311 const SwFieldTypesEnum nType =
312 (nsSwGetSetExpType::GSE_FORMULA & m_nSubType)
313 ? SwFieldTypesEnum::Formel
314 : SwFieldTypesEnum::Get;
316 return SwFieldType::GetTypeStr(nType) + " " + GetFormula();
319 std::unique_ptr<SwField> SwGetExpField::Copy() const
321 std::unique_ptr<SwGetExpField> pTmp(new SwGetExpField(static_cast<SwGetExpFieldType*>(GetTyp()),
322 GetFormula(), m_nSubType, GetFormat()));
323 pTmp->SetLanguage(GetLanguage());
324 pTmp->m_fValueRLHidden = m_fValueRLHidden;
325 pTmp->SwValueField::SetValue(GetValue());
326 pTmp->m_sExpand = m_sExpand;
327 pTmp->m_sExpandRLHidden = m_sExpandRLHidden;
328 pTmp->m_bIsInBodyText = m_bIsInBodyText;
329 pTmp->SetAutomaticLanguage(IsAutomaticLanguage());
330 if( m_bLateInitialization )
331 pTmp->SetLateInitialization();
333 return std::unique_ptr<SwField>(pTmp.release());
336 void SwGetExpField::ChangeExpansion( const SwFrame& rFrame, const SwTextField& rField )
338 if( m_bIsInBodyText ) // only fields in Footer, Header, FootNote, Flys
339 return;
341 OSL_ENSURE( !rFrame.IsInDocBody(), "Flag incorrect, frame is in DocBody" );
343 // determine document (or is there an easier way?)
344 const SwTextNode* pTextNode = &rField.GetTextNode();
345 SwDoc& rDoc = const_cast<SwDoc&>(pTextNode->GetDoc());
347 // create index for determination of the TextNode
348 SwPosition aPos( rDoc.GetNodes() );
349 pTextNode = GetBodyTextNode( rDoc, aPos, rFrame );
351 // If no layout exists, ChangeExpansion is called for header and
352 // footer lines via layout formatting without existing TextNode.
353 if(!pTextNode)
354 return;
355 // #i82544#
356 if( m_bLateInitialization )
358 SwFieldType* pSetExpField = rDoc.getIDocumentFieldsAccess().GetFieldType(SwFieldIds::SetExp, GetFormula(), false);
359 if( pSetExpField )
361 m_bLateInitialization = false;
362 if( !(GetSubType() & nsSwGetSetExpType::GSE_STRING) &&
363 static_cast< SwSetExpFieldType* >(pSetExpField)->GetType() == nsSwGetSetExpType::GSE_STRING )
364 SetSubType( nsSwGetSetExpType::GSE_STRING );
368 SwRootFrame const& rLayout(*rFrame.getRootFrame());
369 OUString & rExpand(rLayout.IsHideRedlines() ? m_sExpandRLHidden : m_sExpand);
370 // here a page number is needed to sort correctly
371 SetGetExpField aEndField(aPos.GetNode(), &rField, aPos.GetContentIndex(), rFrame.GetPhyPageNum());
372 if(GetSubType() & nsSwGetSetExpType::GSE_STRING)
374 SwHashTable<HashStr> aHashTable(0);
375 rDoc.getIDocumentFieldsAccess().FieldsToExpand(aHashTable, aEndField, rLayout);
376 rExpand = LookString( aHashTable, GetFormula() );
378 else
380 // fill calculator with values
381 SwCalc aCalc( rDoc );
382 rDoc.getIDocumentFieldsAccess().FieldsToCalc(aCalc, aEndField, &rLayout);
384 // calculate value
385 SetValue(aCalc.Calculate(GetFormula()).GetDouble(), &rLayout);
387 // analyse based on format
388 rExpand = static_cast<SwValueFieldType*>(GetTyp())->ExpandValue(
389 GetValue(&rLayout), GetFormat(), GetLanguage());
393 OUString SwGetExpField::GetPar2() const
395 return GetFormula();
398 void SwGetExpField::SetPar2(const OUString& rStr)
400 SetFormula(rStr);
403 sal_uInt16 SwGetExpField::GetSubType() const
405 return m_nSubType;
408 void SwGetExpField::SetSubType(sal_uInt16 nType)
410 m_nSubType = nType;
413 void SwGetExpField::SetLanguage(LanguageType nLng)
415 if (m_nSubType & nsSwExtendedSubType::SUB_CMD)
416 SwField::SetLanguage(nLng);
417 else
418 SwValueField::SetLanguage(nLng);
421 bool SwGetExpField::QueryValue( uno::Any& rAny, sal_uInt16 nWhichId ) const
423 switch( nWhichId )
425 case FIELD_PROP_DOUBLE:
426 rAny <<= GetValue();
427 break;
428 case FIELD_PROP_FORMAT:
429 rAny <<= static_cast<sal_Int32>(GetFormat());
430 break;
431 case FIELD_PROP_USHORT1:
432 rAny <<= static_cast<sal_Int16>(m_nSubType);
433 break;
434 case FIELD_PROP_PAR1:
435 rAny <<= GetFormula();
436 break;
437 case FIELD_PROP_SUBTYPE:
439 sal_Int16 nRet = lcl_SubTypeToAPI(GetSubType() & 0xff);
440 rAny <<= nRet;
442 break;
443 case FIELD_PROP_BOOL2:
444 rAny <<= 0 != (m_nSubType & nsSwExtendedSubType::SUB_CMD);
445 break;
446 case FIELD_PROP_PAR4:
447 rAny <<= m_sExpand;
448 break;
449 default:
450 return SwField::QueryValue(rAny, nWhichId);
452 return true;
455 bool SwGetExpField::PutValue( const uno::Any& rAny, sal_uInt16 nWhichId )
457 sal_Int32 nTmp = 0;
458 switch( nWhichId )
460 case FIELD_PROP_DOUBLE:
461 SwValueField::SetValue(*o3tl::doAccess<double>(rAny));
462 m_fValueRLHidden = *o3tl::doAccess<double>(rAny);
463 break;
464 case FIELD_PROP_FORMAT:
465 rAny >>= nTmp;
466 SetFormat(nTmp);
467 break;
468 case FIELD_PROP_USHORT1:
469 rAny >>= nTmp;
470 m_nSubType = o3tl::narrowing<sal_uInt16>(nTmp);
471 break;
472 case FIELD_PROP_PAR1:
474 OUString sTmp;
475 rAny >>= sTmp;
476 SetFormula(sTmp);
477 break;
479 case FIELD_PROP_SUBTYPE:
480 nTmp = lcl_APIToSubType(rAny);
481 if( nTmp >=0 )
482 SetSubType( o3tl::narrowing<sal_uInt16>((GetSubType() & 0xff00) | nTmp));
483 break;
484 case FIELD_PROP_BOOL2:
485 if(*o3tl::doAccess<bool>(rAny))
486 m_nSubType |= nsSwExtendedSubType::SUB_CMD;
487 else
488 m_nSubType &= (~nsSwExtendedSubType::SUB_CMD);
489 break;
490 case FIELD_PROP_PAR4:
492 OUString sTmp;
493 rAny >>= sTmp;
494 ChgExpStr(sTmp, nullptr);
495 break;
497 default:
498 return SwField::PutValue(rAny, nWhichId);
500 return true;
503 SwSetExpFieldType::SwSetExpFieldType( SwDoc* pDc, OUString aName, sal_uInt16 nTyp )
504 : SwValueFieldType( pDc, SwFieldIds::SetExp ),
505 m_sName( std::move(aName) ),
506 m_sDelim( "." ),
507 m_nType(nTyp), m_nLevel( UCHAR_MAX ),
508 m_bDeleted( false )
510 if( ( nsSwGetSetExpType::GSE_SEQ | nsSwGetSetExpType::GSE_STRING ) & m_nType )
511 EnableFormat(false); // do not use Numberformatter
514 std::unique_ptr<SwFieldType> SwSetExpFieldType::Copy() const
516 std::unique_ptr<SwSetExpFieldType> pNew(new SwSetExpFieldType(GetDoc(), m_sName, m_nType));
517 pNew->m_bDeleted = m_bDeleted;
518 pNew->m_sDelim = m_sDelim;
519 pNew->m_nLevel = m_nLevel;
521 return pNew;
524 OUString SwSetExpFieldType::GetName() const
526 return m_sName;
529 const OUString& SwSetExpField::GetExpStr(SwRootFrame const*const pLayout) const
531 return (pLayout && pLayout->IsHideRedlines()) ? msExpandRLHidden : msExpand;
534 void SwSetExpField::ChgExpStr(const OUString& rExpand, SwRootFrame const*const pLayout)
536 if (!pLayout || pLayout->IsHideRedlines())
538 msExpandRLHidden = rExpand;
540 if (!pLayout || !pLayout->IsHideRedlines())
542 msExpand = rExpand;
546 void SwSetExpFieldType::SwClientNotify(const SwModify&, const SfxHint&)
548 // do not expand further
551 void SwSetExpFieldType::SetSeqFormat(sal_uLong nFormat)
553 std::vector<SwFormatField*> vFields;
554 GatherFields(vFields, false);
555 for(auto pFormatField: vFields)
556 pFormatField->GetField()->ChangeFormat(nFormat);
559 sal_uLong SwSetExpFieldType::GetSeqFormat() const
561 if( !HasWriterListeners() )
562 return SVX_NUM_ARABIC;
564 std::vector<SwFormatField*> vFields;
565 GatherFields(vFields, false);
566 return vFields.front()->GetField()->GetFormat();
569 void SwSetExpFieldType::SetSeqRefNo( SwSetExpField& rField )
571 if( !HasWriterListeners() || !(nsSwGetSetExpType::GSE_SEQ & m_nType) )
572 return;
574 std::vector<sal_uInt16> aArr;
576 // check if number is already used and if a new one needs to be created
577 std::vector<SwFormatField*> vFields;
578 GatherFields(vFields);
579 for(SwFormatField* pF: vFields)
580 if(pF->GetField() != &rField)
581 InsertSort(aArr, static_cast<SwSetExpField*>(pF->GetField())->GetSeqNumber());
583 // check first if number already exists
584 sal_uInt16 nNum = rField.GetSeqNumber();
585 if( USHRT_MAX != nNum )
587 std::vector<sal_uInt16>::size_type n {0};
589 for( n = 0; n < aArr.size(); ++n )
590 if( aArr[ n ] >= nNum )
591 break;
593 if( n == aArr.size() || aArr[ n ] > nNum )
594 return; // no -> use it
597 // flagged all numbers, so determine the right number
598 std::vector<sal_uInt16>::size_type n = aArr.size();
599 OSL_ENSURE( n <= std::numeric_limits<sal_uInt16>::max(), "Array is too big for using a sal_uInt16 index" );
601 if ( n > 0 && aArr[ n-1 ] != n-1 )
603 for( n = 0; n < aArr.size(); ++n )
604 if( n != aArr[ n ] )
605 break;
608 rField.SetSeqNumber( n );
611 size_t SwSetExpFieldType::GetSeqFieldList(SwSeqFieldList& rList,
612 SwRootFrame const*const pLayout)
614 rList.Clear();
616 IDocumentRedlineAccess const& rIDRA(GetDoc()->getIDocumentRedlineAccess());
618 std::vector<SwFormatField*> vFields;
619 GatherFields(vFields);
620 for(SwFormatField* pF: vFields)
622 const SwTextNode* pNd;
623 if( nullptr != ( pNd = pF->GetTextField()->GetpTextNode() )
624 && (!pLayout || !pLayout->IsHideRedlines()
625 || !sw::IsFieldDeletedInModel(rIDRA, *pF->GetTextField())))
627 SeqFieldLstElem aNew(
628 pNd->GetExpandText(pLayout),
629 static_cast<SwSetExpField*>(pF->GetField())->GetSeqNumber() );
630 rList.InsertSort( std::move(aNew) );
633 return rList.Count();
636 void SwSetExpFieldType::SetChapter(SwSetExpField& rField, const SwNode& rNd,
637 SwRootFrame const*const pLayout)
639 const SwTextNode* pTextNd = rNd.FindOutlineNodeOfLevel(m_nLevel, pLayout);
640 if( !pTextNd )
641 return;
643 SwNumRule * pRule = pTextNd->GetNumRule();
645 if (!pRule)
646 return;
648 // --> OD 2005-11-02 #i51089 - TUNING#
649 if (SwNodeNum const*const pNum = pTextNd->GetNum(pLayout))
651 // only get the number, without pre-/post-fixstrings
652 OUString const sNumber(pRule->MakeNumString(*pNum, false));
654 if( !sNumber.isEmpty() )
655 rField.ChgExpStr(sNumber + m_sDelim + rField.GetExpStr(pLayout), pLayout);
657 else
659 OSL_ENSURE(pTextNd->GetNum(nullptr), "<SwSetExpFieldType::SetChapter(..)> - text node with numbering rule, but without number. This is a serious defect");
663 void SwSetExpFieldType::QueryValue( uno::Any& rAny, sal_uInt16 nWhichId ) const
665 switch( nWhichId )
667 case FIELD_PROP_SUBTYPE:
669 sal_Int16 nRet = lcl_SubTypeToAPI(GetType());
670 rAny <<= nRet;
672 break;
673 case FIELD_PROP_PAR2:
674 rAny <<= GetDelimiter();
675 break;
676 case FIELD_PROP_SHORT1:
678 sal_Int8 nRet = m_nLevel < MAXLEVEL? m_nLevel : -1;
679 rAny <<= nRet;
681 break;
682 default:
683 assert(false);
687 void SwSetExpFieldType::PutValue( const uno::Any& rAny, sal_uInt16 nWhichId )
689 switch( nWhichId )
691 case FIELD_PROP_SUBTYPE:
693 sal_Int32 nSet = lcl_APIToSubType(rAny);
694 if(nSet >=0)
695 SetType(o3tl::narrowing<sal_uInt16>(nSet));
697 break;
698 case FIELD_PROP_PAR2:
700 OUString sTmp;
701 rAny >>= sTmp;
702 if( !sTmp.isEmpty() )
703 SetDelimiter( sTmp );
704 else
705 SetDelimiter( " " );
707 break;
708 case FIELD_PROP_SHORT1:
710 sal_Int8 nLvl = 0;
711 rAny >>= nLvl;
712 if(nLvl < 0 || nLvl >= MAXLEVEL)
713 SetOutlineLvl(UCHAR_MAX);
714 else
715 SetOutlineLvl(nLvl);
717 break;
718 default:
719 assert(false);
723 bool SwSeqFieldList::InsertSort( SeqFieldLstElem aNew )
725 OUStringBuffer aBuf(aNew.sDlgEntry);
726 const sal_Int32 nLen = aBuf.getLength();
727 for (sal_Int32 i = 0; i < nLen; ++i)
729 if (aBuf[i]<' ')
731 aBuf[i]=' ';
734 aNew.sDlgEntry = aBuf.makeStringAndClear();
736 size_t nPos = 0;
737 bool bRet = SeekEntry( aNew, &nPos );
738 if( !bRet )
739 maData.insert( maData.begin() + nPos, aNew );
740 return bRet;
743 bool SwSeqFieldList::SeekEntry( const SeqFieldLstElem& rNew, size_t* pP ) const
745 size_t nO = maData.size();
746 size_t nU = 0;
747 if( nO > 0 )
749 CollatorWrapper & rCaseColl = ::GetAppCaseCollator(),
750 & rColl = ::GetAppCollator();
751 const CharClass& rCC = GetAppCharClass();
753 //#59900# Sorting should sort number correctly (e.g. "10" after "9" not after "1")
754 const OUString rTmp2 = rNew.sDlgEntry;
755 sal_Int32 nFndPos2 = 0;
756 const OUString sNum2( rTmp2.getToken( 0, ' ', nFndPos2 ));
757 bool bIsNum2IsNumeric = CharClass::isAsciiNumeric( sNum2 );
758 sal_Int32 nNum2 = bIsNum2IsNumeric ? sNum2.toInt32() : 0;
760 nO--;
761 while( nU <= nO )
763 const size_t nM = nU + ( nO - nU ) / 2;
765 //#59900# Sorting should sort number correctly (e.g. "10" after "9" not after "1")
766 const OUString rTmp1 = maData[nM].sDlgEntry;
767 sal_Int32 nFndPos1 = 0;
768 const OUString sNum1( rTmp1.getToken( 0, ' ', nFndPos1 ));
769 sal_Int32 nCmp;
771 if( bIsNum2IsNumeric && rCC.isNumeric( sNum1 ) )
773 sal_Int32 nNum1 = sNum1.toInt32();
774 nCmp = nNum2 - nNum1;
775 if( 0 == nCmp )
777 OUString aTmp1 = nFndPos1 != -1 ? rTmp1.copy(nFndPos1) : OUString();
778 OUString aTmp2 = nFndPos2 != -1 ? rTmp2.copy(nFndPos2) : OUString();
779 nCmp = rCaseColl.compareString(aTmp2, aTmp1);
782 else
783 nCmp = rColl.compareString( rTmp2, rTmp1 );
785 if( 0 == nCmp )
787 if( pP ) *pP = nM;
788 return true;
790 else if( 0 < nCmp )
791 nU = nM + 1;
792 else if( nM == 0 )
793 break;
794 else
795 nO = nM - 1;
798 if( pP ) *pP = nU;
799 return false;
802 SwSetExpField::SwSetExpField(SwSetExpFieldType* pTyp, const OUString& rFormel,
803 sal_uLong nFormat)
804 : SwFormulaField( pTyp, nFormat, 0.0 )
805 , m_fValueRLHidden(0.0)
806 , mnSeqNo( USHRT_MAX )
807 , mnSubType(0)
808 , mpFormatField(nullptr)
810 SetFormula(rFormel);
811 // ignore SubType
812 mbInput = false;
813 if( IsSequenceField() )
815 SwValueField::SetValue(1.0);
816 m_fValueRLHidden = 1.0;
817 if( rFormel.isEmpty() )
819 SetFormula(pTyp->GetName() + "+1");
824 void SwSetExpField::SetFormatField(SwFormatField & rFormatField)
826 mpFormatField = &rFormatField;
829 OUString SwSetExpField::ExpandImpl(SwRootFrame const*const pLayout) const
831 if (mnSubType & nsSwExtendedSubType::SUB_CMD)
832 { // we need the CommandString
833 return GetTyp()->GetName() + " = " + GetFormula();
835 if(!(mnSubType & nsSwExtendedSubType::SUB_INVISIBLE))
836 { // value is visible
837 return (pLayout && pLayout->IsHideRedlines()) ? msExpandRLHidden : msExpand;
839 return OUString();
842 /// @return the field name
843 OUString SwSetExpField::GetFieldName() const
845 SwFieldTypesEnum const nStrType( (IsSequenceField())
846 ? SwFieldTypesEnum::Sequence
847 : mbInput
848 ? SwFieldTypesEnum::SetInput
849 : SwFieldTypesEnum::Set );
851 OUString aStr(
852 SwFieldType::GetTypeStr( nStrType )
853 + " "
854 + GetTyp()->GetName() );
856 // Sequence: without formula
857 if (SwFieldTypesEnum::Sequence != nStrType)
859 aStr += " = " + GetFormula();
861 return aStr;
864 std::unique_ptr<SwField> SwSetExpField::Copy() const
866 std::unique_ptr<SwSetExpField> pTmp(new SwSetExpField(static_cast<SwSetExpFieldType*>(GetTyp()),
867 GetFormula(), GetFormat()));
868 pTmp->SwValueField::SetValue(GetValue());
869 pTmp->m_fValueRLHidden = m_fValueRLHidden;
870 pTmp->msExpand = msExpand;
871 pTmp->msExpandRLHidden = msExpandRLHidden;
872 pTmp->SetAutomaticLanguage(IsAutomaticLanguage());
873 pTmp->SetLanguage(GetLanguage());
874 pTmp->maPText = maPText;
875 pTmp->mbInput = mbInput;
876 pTmp->mnSeqNo = mnSeqNo;
877 pTmp->SetSubType(GetSubType());
879 return std::unique_ptr<SwField>(pTmp.release());
882 void SwSetExpField::SetSubType(sal_uInt16 nSub)
884 static_cast<SwSetExpFieldType*>(GetTyp())->SetType(nSub & 0xff);
885 mnSubType = nSub & 0xff00;
887 OSL_ENSURE( (nSub & 0xff) != 3, "SubType is illegal!" );
890 sal_uInt16 SwSetExpField::GetSubType() const
892 return static_cast<SwSetExpFieldType*>(GetTyp())->GetType() | mnSubType;
895 void SwSetExpField::SetValue( const double& rAny )
897 SwValueField::SetValue(rAny);
899 if( IsSequenceField() )
900 msExpand = FormatNumber( GetValue(), static_cast<SvxNumType>(GetFormat()), GetLanguage() );
901 else
902 msExpand = static_cast<SwValueFieldType*>(GetTyp())->ExpandValue( rAny,
903 GetFormat(), GetLanguage());
906 void SwSetExpField::SetValue(const double& rValue, SwRootFrame const*const pLayout)
908 if (!pLayout || !pLayout->IsHideRedlines())
910 SetValue(rValue);
912 if (pLayout && !pLayout->IsHideRedlines())
913 return;
915 m_fValueRLHidden = rValue;
916 if (IsSequenceField())
918 msExpandRLHidden = FormatNumber(rValue, static_cast<SvxNumType>(GetFormat()), GetLanguage());
920 else
922 msExpandRLHidden = static_cast<SwValueFieldType*>(GetTyp())->ExpandValue(
923 rValue, GetFormat(), GetLanguage());
927 double SwSetExpField::GetValue(SwRootFrame const* pLayout) const
929 return (pLayout && pLayout->IsHideRedlines()) ? m_fValueRLHidden : GetValue();
932 void SwGetExpField::SetValue( const double& rAny )
934 SwValueField::SetValue(rAny);
935 m_sExpand = static_cast<SwValueFieldType*>(GetTyp())->ExpandValue( rAny, GetFormat(),
936 GetLanguage());
939 void SwGetExpField::SetValue(const double& rValue, SwRootFrame const*const pLayout)
941 if (!pLayout || !pLayout->IsHideRedlines())
943 SetValue(rValue);
945 if (!pLayout || pLayout->IsHideRedlines())
947 m_fValueRLHidden = rValue;
948 m_sExpandRLHidden = static_cast<SwValueFieldType*>(GetTyp())->ExpandValue(
949 rValue, GetFormat(), GetLanguage());
953 double SwGetExpField::GetValue(SwRootFrame const* pLayout) const
955 return (pLayout && pLayout->IsHideRedlines()) ? m_fValueRLHidden : GetValue();
958 /** Find the index of the reference text following the current field
960 * @param rFormat
961 * @param rDoc
962 * @param nHint search starting position after the current field (or 0 if default)
963 * @return
965 sal_Int32 SwGetExpField::GetReferenceTextPos( const SwFormatField& rFormat, SwDoc& rDoc, sal_Int32 nHint)
968 const SwTextField* pTextField = rFormat.GetTextField();
969 const SwTextNode& rTextNode = pTextField->GetTextNode();
971 sal_Int32 nRet = nHint ? nHint : pTextField->GetStart() + 1;
972 OUString sNodeText = rTextNode.GetText();
974 if(nRet<sNodeText.getLength())
976 sNodeText = sNodeText.copy(nRet);
978 // now check if sNodeText starts with a non-alphanumeric character plus blanks
979 sal_uInt16 nSrcpt = g_pBreakIt->GetRealScriptOfText( sNodeText, 0 );
981 static const WhichRangesContainer nIds(svl::Items<
982 RES_CHRATR_FONT, RES_CHRATR_FONT,
983 RES_CHRATR_LANGUAGE, RES_CHRATR_LANGUAGE,
984 RES_CHRATR_CJK_FONT, RES_CHRATR_CJK_FONT,
985 RES_CHRATR_CJK_LANGUAGE, RES_CHRATR_CJK_LANGUAGE,
986 RES_CHRATR_CTL_FONT, RES_CHRATR_CTL_FONT,
987 RES_CHRATR_CTL_LANGUAGE, RES_CHRATR_CTL_LANGUAGE
989 SwAttrSet aSet(rDoc.GetAttrPool(), nIds);
990 rTextNode.GetParaAttr(aSet, nRet, nRet+1);
992 TypedWhichId<SvxFontItem> nFontWhich = GetWhichOfScript( RES_CHRATR_FONT, nSrcpt );
993 if( RTL_TEXTENCODING_SYMBOL != aSet.Get( nFontWhich ).GetCharSet() )
995 TypedWhichId<SvxLanguageItem> nLangWhich = GetWhichOfScript( RES_CHRATR_LANGUAGE, nSrcpt ) ;
996 LanguageType eLang = aSet.Get(nLangWhich).GetLanguage();
997 CharClass aCC(( LanguageTag(eLang) ));
998 sal_Unicode c0 = sNodeText[0];
999 bool bIsAlphaNum = aCC.isAlphaNumeric( sNodeText, 0 );
1000 if( !bIsAlphaNum ||
1001 (c0 == ' ' || c0 == '\t'))
1003 // ignoring blanks
1004 nRet++;
1005 const sal_Int32 nLen = sNodeText.getLength();
1006 for (sal_Int32 i = 1;
1007 i<nLen && (sNodeText[i]==' ' || sNodeText[i]=='\t');
1010 ++nRet;
1014 return nRet;
1017 OUString SwSetExpField::GetPar1() const
1019 return static_cast<const SwSetExpFieldType*>(GetTyp())->GetName();
1022 OUString SwSetExpField::GetPar2() const
1024 sal_uInt16 nType = static_cast<SwSetExpFieldType*>(GetTyp())->GetType();
1026 if (nType & nsSwGetSetExpType::GSE_STRING)
1027 return GetFormula();
1028 return GetExpandedFormula();
1031 void SwSetExpField::SetPar2(const OUString& rStr)
1033 sal_uInt16 nType = static_cast<SwSetExpFieldType*>(GetTyp())->GetType();
1035 if( !(nType & nsSwGetSetExpType::GSE_SEQ) || !rStr.isEmpty() )
1037 if (nType & nsSwGetSetExpType::GSE_STRING)
1038 SetFormula(rStr);
1039 else
1040 SetExpandedFormula(rStr);
1044 bool SwSetExpField::PutValue( const uno::Any& rAny, sal_uInt16 nWhichId )
1046 sal_Int32 nTmp32 = 0;
1047 sal_Int16 nTmp16 = 0;
1048 switch( nWhichId )
1050 case FIELD_PROP_BOOL2:
1051 if(*o3tl::doAccess<bool>(rAny))
1052 mnSubType &= ~nsSwExtendedSubType::SUB_INVISIBLE;
1053 else
1054 mnSubType |= nsSwExtendedSubType::SUB_INVISIBLE;
1055 break;
1056 case FIELD_PROP_FORMAT:
1057 rAny >>= nTmp32;
1058 SetFormat(nTmp32);
1059 break;
1060 case FIELD_PROP_USHORT2:
1062 rAny >>= nTmp16;
1063 if(nTmp16 <= css::style::NumberingType::NUMBER_NONE )
1064 SetFormat(nTmp16);
1065 else {
1066 //exception(wrong_value)
1070 break;
1071 case FIELD_PROP_USHORT1:
1072 rAny >>= nTmp16;
1073 mnSeqNo = nTmp16;
1074 break;
1075 case FIELD_PROP_PAR1:
1077 OUString sTmp;
1078 rAny >>= sTmp;
1079 SetPar1( SwStyleNameMapper::GetUIName( sTmp, SwGetPoolIdFromName::TxtColl ) );
1081 break;
1082 case FIELD_PROP_PAR2:
1084 OUString uTmp;
1085 rAny >>= uTmp;
1086 //I18N - if the formula contains only "TypeName+1"
1087 //and it's one of the initially created sequence fields
1088 //then the localized names has to be replaced by a programmatic name
1089 OUString sMyFormula = SwXFieldMaster::LocalizeFormula(*this, uTmp, false);
1090 SetFormula( sMyFormula );
1092 break;
1093 case FIELD_PROP_DOUBLE:
1095 double fVal = 0.0;
1096 rAny >>= fVal;
1097 SetValue(fVal);
1098 m_fValueRLHidden = fVal;
1100 break;
1101 case FIELD_PROP_SUBTYPE:
1102 nTmp32 = lcl_APIToSubType(rAny);
1103 if(nTmp32 >= 0)
1104 SetSubType(o3tl::narrowing<sal_uInt16>((GetSubType() & 0xff00) | nTmp32));
1105 break;
1106 case FIELD_PROP_PAR3:
1107 rAny >>= maPText;
1108 break;
1109 case FIELD_PROP_BOOL3:
1110 if(*o3tl::doAccess<bool>(rAny))
1111 mnSubType |= nsSwExtendedSubType::SUB_CMD;
1112 else
1113 mnSubType &= (~nsSwExtendedSubType::SUB_CMD);
1114 break;
1115 case FIELD_PROP_BOOL1:
1117 bool newInput(*o3tl::doAccess<bool>(rAny));
1118 if (newInput != GetInputFlag())
1120 if (static_cast<SwSetExpFieldType*>(GetTyp())->GetType()
1121 & nsSwGetSetExpType::GSE_STRING)
1123 SwXTextField::TransmuteLeadToInputField(*this);
1125 else
1127 SetInputFlag(newInput);
1131 break;
1132 case FIELD_PROP_PAR4:
1134 OUString sTmp;
1135 rAny >>= sTmp;
1136 ChgExpStr(sTmp, nullptr);
1138 break;
1139 default:
1140 return SwField::PutValue(rAny, nWhichId);
1142 return true;
1145 bool SwSetExpField::QueryValue( uno::Any& rAny, sal_uInt16 nWhichId ) const
1147 switch( nWhichId )
1149 case FIELD_PROP_BOOL2:
1150 rAny <<= 0 == (mnSubType & nsSwExtendedSubType::SUB_INVISIBLE);
1151 break;
1152 case FIELD_PROP_FORMAT:
1153 rAny <<= static_cast<sal_Int32>(GetFormat());
1154 break;
1155 case FIELD_PROP_USHORT2:
1156 rAny <<= static_cast<sal_Int16>(GetFormat());
1157 break;
1158 case FIELD_PROP_USHORT1:
1159 rAny <<= static_cast<sal_Int16>(mnSeqNo);
1160 break;
1161 case FIELD_PROP_PAR1:
1162 rAny <<= SwStyleNameMapper::GetProgName(GetPar1(), SwGetPoolIdFromName::TxtColl );
1163 break;
1164 case FIELD_PROP_PAR2:
1166 //I18N - if the formula contains only "TypeName+1"
1167 //and it's one of the initially created sequence fields
1168 //then the localized names has to be replaced by a programmatic name
1169 OUString sMyFormula = SwXFieldMaster::LocalizeFormula(*this, GetFormula(), true);
1170 rAny <<= sMyFormula;
1172 break;
1173 case FIELD_PROP_DOUBLE:
1174 rAny <<= GetValue();
1175 break;
1176 case FIELD_PROP_SUBTYPE:
1178 sal_Int16 nRet = lcl_SubTypeToAPI(GetSubType() & 0xff);
1179 rAny <<= nRet;
1181 break;
1182 case FIELD_PROP_PAR3:
1183 rAny <<= maPText;
1184 break;
1185 case FIELD_PROP_BOOL3:
1186 rAny <<= 0 != (mnSubType & nsSwExtendedSubType::SUB_CMD);
1187 break;
1188 case FIELD_PROP_BOOL1:
1189 rAny <<= GetInputFlag();
1190 break;
1191 case FIELD_PROP_PAR4:
1192 rAny <<= GetExpStr(nullptr);
1193 break;
1194 default:
1195 return SwField::QueryValue(rAny, nWhichId);
1197 return true;
1200 SwInputFieldType::SwInputFieldType( SwDoc* pD )
1201 : SwFieldType( SwFieldIds::Input )
1202 , mpDoc( pD )
1206 std::unique_ptr<SwFieldType> SwInputFieldType::Copy() const
1208 return std::make_unique<SwInputFieldType>( mpDoc );
1211 SwInputField::SwInputField( SwInputFieldType* pFieldType,
1212 OUString aContent,
1213 OUString aPrompt,
1214 sal_uInt16 nSub,
1215 sal_uLong nFormat,
1216 bool bIsFormField )
1217 : SwField( pFieldType, nFormat, LANGUAGE_SYSTEM, false )
1218 , maContent(std::move(aContent))
1219 , maPText(std::move(aPrompt))
1220 , mnSubType(nSub)
1221 , mbIsFormField( bIsFormField )
1222 , mpFormatField( nullptr )
1226 SwInputField::~SwInputField()
1230 void SwInputField::SetFormatField( SwFormatField& rFormatField )
1232 mpFormatField = &rFormatField;
1236 void SwInputField::applyFieldContent( const OUString& rNewFieldContent )
1238 if ( (mnSubType & 0x00ff) == INP_TXT )
1240 maContent = rNewFieldContent;
1242 else if( (mnSubType & 0x00ff) == INP_USR )
1244 SwUserFieldType* pUserTyp = static_cast<SwUserFieldType*>(
1245 static_cast<SwInputFieldType*>(GetTyp())->GetDoc()->getIDocumentFieldsAccess().GetFieldType( SwFieldIds::User, getContent(), false ) );
1246 if( pUserTyp )
1248 pUserTyp->SetContent( rNewFieldContent );
1249 if (!pUserTyp->IsModifyLocked())
1251 // trigger update of the corresponding User Fields and other
1252 // related Input Fields
1253 bool bUnlock(false);
1254 if (GetFormatField() != nullptr)
1256 SwTextInputField *const pTextInputField =
1257 dynamic_cast<SwTextInputField*>(GetFormatField()->GetTextField());
1258 if (pTextInputField != nullptr)
1260 bUnlock = pTextInputField->LockNotifyContentChange();
1263 pUserTyp->UpdateFields();
1264 if (bUnlock)
1266 SwTextInputField *const pTextInputField =
1267 dynamic_cast<SwTextInputField*>(GetFormatField()->GetTextField());
1268 if (pTextInputField != nullptr)
1270 pTextInputField->UnlockNotifyContentChange();
1278 OUString SwInputField::GetFieldName() const
1280 OUString aStr(SwField::GetFieldName());
1281 if ((mnSubType & 0x00ff) == INP_USR)
1283 aStr += GetTyp()->GetName() + " " + getContent();
1285 return aStr;
1288 std::unique_ptr<SwField> SwInputField::Copy() const
1290 std::unique_ptr<SwInputField> pField(
1291 new SwInputField(
1292 static_cast<SwInputFieldType*>(GetTyp()),
1293 getContent(),
1294 maPText,
1295 GetSubType(),
1296 GetFormat(),
1297 mbIsFormField ));
1299 pField->SetHelp( maHelp );
1300 pField->SetToolTip( maToolTip );
1301 pField->maGrabBag = maGrabBag;
1303 pField->SetAutomaticLanguage(IsAutomaticLanguage());
1304 return std::unique_ptr<SwField>(pField.release());
1307 OUString SwInputField::ExpandImpl(SwRootFrame const*const) const
1309 if((mnSubType & 0x00ff) == INP_TXT)
1311 return getContent();
1314 if( (mnSubType & 0x00ff) == INP_USR )
1316 SwUserFieldType* pUserTyp = static_cast<SwUserFieldType*>(
1317 static_cast<SwInputFieldType*>(GetTyp())->GetDoc()->getIDocumentFieldsAccess().GetFieldType( SwFieldIds::User, getContent(), false ) );
1318 if( pUserTyp )
1319 return pUserTyp->GetContent();
1322 return OUString();
1325 bool SwInputField::isFormField() const
1327 return mbIsFormField
1328 || !maHelp.isEmpty()
1329 || !maToolTip.isEmpty();
1332 bool SwInputField::QueryValue( uno::Any& rAny, sal_uInt16 nWhichId ) const
1334 switch( nWhichId )
1336 case FIELD_PROP_PAR1:
1337 rAny <<= getContent();
1338 break;
1339 case FIELD_PROP_PAR2:
1340 rAny <<= maPText;
1341 break;
1342 case FIELD_PROP_PAR3:
1343 rAny <<= maHelp;
1344 break;
1345 case FIELD_PROP_PAR4:
1346 rAny <<= maToolTip;
1347 break;
1348 case FIELD_PROP_GRABBAG:
1349 rAny <<= maGrabBag;
1350 break;
1351 case FIELD_PROP_TITLE:
1352 break;
1353 default:
1354 assert(false);
1356 return true;
1359 bool SwInputField::PutValue( const uno::Any& rAny, sal_uInt16 nWhichId )
1361 switch( nWhichId )
1363 case FIELD_PROP_PAR1:
1364 rAny >>= maContent;
1365 break;
1366 case FIELD_PROP_PAR2:
1367 rAny >>= maPText;
1368 break;
1369 case FIELD_PROP_PAR3:
1370 rAny >>= maHelp;
1371 break;
1372 case FIELD_PROP_PAR4:
1373 rAny >>= maToolTip;
1374 break;
1375 case FIELD_PROP_GRABBAG:
1376 rAny >>= maGrabBag;
1377 break;
1378 case FIELD_PROP_TITLE:
1379 break;
1380 default:
1381 assert(false);
1383 return true;
1386 /// set condition
1387 void SwInputField::SetPar1(const OUString& rStr)
1389 maContent = rStr;
1392 OUString SwInputField::GetPar1() const
1394 return getContent();
1397 void SwInputField::SetPar2(const OUString& rStr)
1399 maPText = rStr;
1402 OUString SwInputField::GetPar2() const
1404 return maPText;
1407 void SwInputField::SetHelp(const OUString & rStr)
1409 maHelp = rStr;
1412 const OUString& SwInputField::GetHelp() const
1414 return maHelp;
1417 void SwInputField::SetToolTip(const OUString & rStr)
1419 maToolTip = rStr;
1422 const OUString& SwInputField::GetToolTip() const
1424 return maToolTip;
1427 sal_uInt16 SwInputField::GetSubType() const
1429 return mnSubType;
1432 void SwInputField::SetSubType(sal_uInt16 nSub)
1434 mnSubType = nSub;
1437 /* vim:set shiftwidth=4 softtabstop=4 expandtab: */