bump product version to 7.6.3.2-android
[LibreOffice.git] / vcl / source / edit / xtextedt.cxx
blob106ad69bf425d5495548f8982da4dc9183eba547
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 <i18nutil/searchopt.hxx>
21 #include <i18nlangtag/languagetag.hxx>
22 #include <vcl/textdata.hxx>
23 #include <vcl/xtextedt.hxx>
24 #include <vcl/svapp.hxx>
25 #include <vcl/settings.hxx>
26 #include <unotools/textsearch.hxx>
27 #include <com/sun/star/util/SearchFlags.hpp>
29 using namespace ::com::sun::star;
31 const std::wstring gaGroupChars = L"(){}[]";
33 ExtTextEngine::ExtTextEngine()
37 ExtTextEngine::~ExtTextEngine()
41 TextSelection ExtTextEngine::MatchGroup( const TextPaM& rCursor ) const
43 TextSelection aSel( rCursor );
44 const sal_Int32 nPos = rCursor.GetIndex();
45 sal_uInt32 nPara = rCursor.GetPara();
46 const sal_uInt32 nParas = GetParagraphCount();
47 if ( ( nPara < nParas ) && ( nPos < GetTextLen( nPara ) ) )
49 size_t nMatchIndex = gaGroupChars.find( GetText( rCursor.GetPara() )[ nPos ] );
50 if ( nMatchIndex != std::wstring::npos )
52 if ( ( nMatchIndex % 2 ) == 0 )
54 // search forwards
55 sal_Unicode nSC = gaGroupChars[ nMatchIndex ];
56 sal_Unicode nEC = gaGroupChars[ nMatchIndex+1 ];
58 sal_Int32 nCur = nPos+1;
59 sal_uInt16 nLevel = 1;
60 while ( nLevel && ( nPara < nParas ) )
62 OUString aStr = GetText( nPara );
63 while ( nCur < aStr.getLength() )
65 if ( aStr[nCur] == nSC )
66 nLevel++;
67 else if ( aStr[nCur] == nEC )
69 nLevel--;
70 if ( !nLevel )
71 break; // while nCur...
73 nCur++;
76 if ( nLevel )
78 nPara++;
79 nCur = 0;
82 if ( nLevel == 0 ) // found
84 aSel.GetStart() = rCursor;
85 aSel.GetEnd() = TextPaM( nPara, nCur+1 );
88 else
90 // search backwards
91 sal_Unicode nEC = gaGroupChars[ nMatchIndex ];
92 sal_Unicode nSC = gaGroupChars[ nMatchIndex-1 ];
94 sal_Int32 nCur = rCursor.GetIndex()-1;
95 sal_uInt16 nLevel = 1;
96 while ( nLevel )
98 if ( GetTextLen( nPara ) )
100 OUString aStr = GetText( nPara );
101 while ( nCur )
103 if ( aStr[nCur] == nSC )
105 nLevel--;
106 if ( !nLevel )
107 break; // while nCur...
109 else if ( aStr[nCur] == nEC )
110 nLevel++;
112 nCur--;
116 if ( nLevel )
118 if ( nPara )
120 nPara--;
121 nCur = GetTextLen( nPara )-1; // no matter if negative, as if Len()
123 else
124 break;
128 if ( nLevel == 0 ) // found
130 aSel.GetStart() = rCursor;
131 ++aSel.GetStart().GetIndex(); // behind the char
132 aSel.GetEnd() = TextPaM( nPara, nCur );
137 return aSel;
140 bool ExtTextEngine::Search( TextSelection& rSel, const i18nutil::SearchOptions& rSearchOptions, bool bForward ) const
142 TextSelection aSel( rSel );
143 aSel.Justify();
145 bool bSearchInSelection = (0 != (rSearchOptions.searchFlag & util::SearchFlags::REG_NOT_BEGINOFLINE) );
147 TextPaM aStartPaM( aSel.GetEnd() );
148 if ( aSel.HasRange() && ( ( bSearchInSelection && bForward ) || ( !bSearchInSelection && !bForward ) ) )
150 aStartPaM = aSel.GetStart();
153 bool bFound = false;
154 sal_uInt32 nEndNode;
156 if ( bSearchInSelection )
157 nEndNode = bForward ? aSel.GetEnd().GetPara() : aSel.GetStart().GetPara();
158 else
159 nEndNode = bForward ? (GetParagraphCount()-1) : 0;
161 const sal_uInt32 nStartNode = aStartPaM.GetPara();
163 i18nutil::SearchOptions aOptions( rSearchOptions );
164 aOptions.Locale = Application::GetSettings().GetLanguageTag().getLocale();
165 utl::TextSearch aSearcher( utl::TextSearch::UpgradeToSearchOptions2(aOptions) );
167 // iterate over the paragraphs
168 for ( sal_uInt32 nNode = nStartNode;
169 bForward ? ( nNode <= nEndNode) : ( nNode >= nEndNode );
170 bForward ? nNode++ : nNode-- )
172 OUString aText = GetText( nNode );
173 sal_Int32 nStartPos = 0;
174 sal_Int32 nEndPos = aText.getLength();
175 if ( nNode == nStartNode )
177 if ( bForward )
178 nStartPos = aStartPaM.GetIndex();
179 else
180 nEndPos = aStartPaM.GetIndex();
182 if ( ( nNode == nEndNode ) && bSearchInSelection )
184 if ( bForward )
185 nEndPos = aSel.GetEnd().GetIndex();
186 else
187 nStartPos = aSel.GetStart().GetIndex();
190 if ( bForward )
191 bFound = aSearcher.SearchForward( aText, &nStartPos, &nEndPos );
192 else
193 bFound = aSearcher.SearchBackward( aText, &nEndPos, &nStartPos );
195 if ( bFound )
197 rSel.GetStart().GetPara() = nNode;
198 rSel.GetStart().GetIndex() = nStartPos;
199 rSel.GetEnd().GetPara() = nNode;
200 rSel.GetEnd().GetIndex() = nEndPos;
201 // Select over the paragraph?
202 // FIXME This should be max long...
203 if( nEndPos == -1)
205 if ( (rSel.GetEnd().GetPara()+1) < GetParagraphCount() )
207 rSel.GetEnd().GetPara()++;
208 rSel.GetEnd().GetIndex() = 0;
210 else
212 rSel.GetEnd().GetIndex() = nStartPos;
213 bFound = false;
217 break;
220 if ( !bForward && !nNode ) // if searching backwards, if nEndNode == 0:
221 break;
224 return bFound;
227 /* vim:set shiftwidth=4 softtabstop=4 expandtab: */