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 <txatritr.hxx>
21
22 #include <com/sun/star/i18n/ScriptType.hpp>
23 #include <com/sun/star/i18n/XBreakIterator.hpp>
24 #include <fchrfmt.hxx>
25 #include <charfmt.hxx>
26 #include <breakit.hxx>
27 #include <ndtxt.hxx>
28 #include <txatbase.hxx>
29
30 using namespace ::com::sun::star;
31
SwScriptIterator(const OUString & rStr,sal_Int32 nStt,bool const bFrwrd)32 SwScriptIterator::SwScriptIterator(
33 const OUString& rStr, sal_Int32 nStt, bool const bFrwrd)
34 : m_rText(rStr)
35 , m_nChgPos(rStr.getLength())
36 , m_nCurScript(i18n::ScriptType::WEAK)
37 , m_bForward(bFrwrd)
38 {
39 assert(g_pBreakIt && g_pBreakIt->GetBreakIter().is());
40 if ( ! bFrwrd && nStt )
41 --nStt;
42
43 sal_Int32 nPos = nStt;
44 m_nCurScript = g_pBreakIt->GetBreakIter()->getScriptType(m_rText, nPos);
45 if( i18n::ScriptType::WEAK == m_nCurScript )
46 {
47 if( nPos )
48 {
49 nPos = g_pBreakIt->GetBreakIter()->beginOfScript(
50 m_rText, nPos, m_nCurScript);
51 if (nPos > 0 && nPos < m_rText.getLength())
52 {
53 nStt = --nPos;
54 m_nCurScript =
55 g_pBreakIt->GetBreakIter()->getScriptType(m_rText,nPos);
56 }
57 }
58 }
59
60 m_nChgPos = m_bForward
61 ? g_pBreakIt->GetBreakIter()->endOfScript(
62 m_rText, nStt, m_nCurScript)
63 : g_pBreakIt->GetBreakIter()->beginOfScript(
64 m_rText, nStt, m_nCurScript);
65 }
66
Next()67 void SwScriptIterator::Next()
68 {
69 assert(g_pBreakIt && g_pBreakIt->GetBreakIter().is());
70 if (m_bForward && m_nChgPos >= 0 && m_nChgPos < m_rText.getLength())
71 {
72 m_nCurScript =
73 g_pBreakIt->GetBreakIter()->getScriptType(m_rText, m_nChgPos);
74 m_nChgPos = g_pBreakIt->GetBreakIter()->endOfScript(
75 m_rText, m_nChgPos, m_nCurScript);
76 }
77 else if (!m_bForward && m_nChgPos > 0)
78 {
79 --m_nChgPos;
80 m_nCurScript =
81 g_pBreakIt->GetBreakIter()->getScriptType(m_rText, m_nChgPos);
82 m_nChgPos = g_pBreakIt->GetBreakIter()->beginOfScript(
83 m_rText, m_nChgPos, m_nCurScript);
84 }
85 }
86
SwLanguageIterator(const SwTextNode & rTNd,sal_Int32 nStt)87 SwLanguageIterator::SwLanguageIterator( const SwTextNode& rTNd,
88 sal_Int32 nStt )
89 : m_aScriptIter( rTNd.GetText(), nStt ),
90 m_rTextNode( rTNd ),
91 m_pParaItem( nullptr ),
92 m_nAttrPos( 0 ),
93 m_nChgPos( nStt )
94 {
95 SearchNextChg();
96 }
97
Next()98 bool SwLanguageIterator::Next()
99 {
100 bool bRet = false;
101 if (m_nChgPos < m_aScriptIter.GetText().getLength())
102 {
103 bRet = true;
104 if( !m_aStack.empty() )
105 {
106 do {
107 const SwTextAttr* pHt = m_aStack.front();
108 const sal_Int32 nEndPos = *pHt->End();
109 if( m_nChgPos >= nEndPos )
110 m_aStack.pop_front();
111 else
112 break;
113 } while( !m_aStack.empty() );
114 }
115
116 if( !m_aStack.empty() )
117 {
118 const size_t nSavePos = m_nAttrPos;
119 SearchNextChg();
120 if( !m_aStack.empty() )
121 {
122 const SwTextAttr* pHt = m_aStack.front();
123 const sal_Int32 nEndPos = *pHt->End();
124 if( m_nChgPos >= nEndPos )
125 {
126 m_nChgPos = nEndPos;
127 m_nAttrPos = nSavePos;
128
129 if( RES_TXTATR_CHARFMT == pHt->Which() )
130 {
131 const sal_uInt16 nWId = GetWhichOfScript( RES_CHRATR_LANGUAGE, m_aScriptIter.GetCurrScript() );
132 m_pCurrentItem = &pHt->GetCharFormat().GetCharFormat()->GetFormatAttr(nWId);
133 }
134 else
135 m_pCurrentItem = &pHt->GetAttr();
136
137 m_aStack.pop_front();
138 }
139 }
140 }
141 else
142 SearchNextChg();
143 }
144 return bRet;
145 }
146
AddToStack(const SwTextAttr & rAttr)147 void SwLanguageIterator::AddToStack( const SwTextAttr& rAttr )
148 {
149 size_t nIns = 0;
150 const sal_Int32 nEndPos = *rAttr.End();
151 for( ; nIns < m_aStack.size(); ++nIns )
152 if( *m_aStack[ nIns ]->End() > nEndPos )
153 break;
154
155 m_aStack.insert( m_aStack.begin() + nIns, &rAttr );
156 }
157
SearchNextChg()158 void SwLanguageIterator::SearchNextChg()
159 {
160 sal_uInt16 nWh = 0;
161 if( m_nChgPos == m_aScriptIter.GetScriptChgPos() )
162 {
163 m_aScriptIter.Next();
164 m_pParaItem = nullptr;
165 m_nAttrPos = 0; // must be restart at the beginning, because
166 // some attributes can start before or inside
167 // the current scripttype!
168 m_aStack.clear();
169 }
170 if( !m_pParaItem )
171 {
172 nWh = GetWhichOfScript( RES_CHRATR_LANGUAGE, m_aScriptIter.GetCurrScript() );
173 m_pParaItem = &m_rTextNode.GetSwAttrSet().Get( nWh );
174 }
175
176 sal_Int32 nStt = m_nChgPos;
177 m_nChgPos = m_aScriptIter.GetScriptChgPos();
178 m_pCurrentItem = m_pParaItem;
179
180 const SwpHints* pHts = m_rTextNode.GetpSwpHints();
181 if( !pHts )
182 return;
183
184 if( !nWh )
185 {
186 nWh = GetWhichOfScript( RES_CHRATR_LANGUAGE, m_aScriptIter.GetCurrScript() );
187 }
188
189 const SfxPoolItem* pItem = nullptr;
190 for( ; m_nAttrPos < pHts->Count(); ++m_nAttrPos )
191 {
192 const SwTextAttr* pHt = pHts->Get( m_nAttrPos );
193 const sal_Int32* pEnd = pHt->End();
194 const sal_Int32 nHtStt = pHt->GetStart();
195 if( nHtStt < nStt && ( !pEnd || *pEnd <= nStt ))
196 continue;
197
198 if( nHtStt >= m_nChgPos )
199 break;
200
201 pItem = CharFormat::GetItem( *pHt, nWh );
202 if ( pItem )
203 {
204 if( nHtStt > nStt )
205 {
206 if( m_nChgPos > nHtStt )
207 m_nChgPos = nHtStt;
208 break;
209 }
210 AddToStack( *pHt );
211 m_pCurrentItem = pItem;
212 if( *pEnd < m_nChgPos )
213 m_nChgPos = *pEnd;
214 }
215 }
216 }
217
218 /* vim:set shiftwidth=4 softtabstop=4 expandtab: */
219