1 // © 2016 and later: Unicode, Inc. and others.
2 // License & terms of use: http://www.unicode.org/copyright.html
4 *******************************************************************************
6 * Copyright (C) 2001-2011, International Business Machines
7 * Corporation and others. All Rights Reserved.
9 *******************************************************************************
10 * file name: casetrn.cpp
12 * tab size: 8 (not used)
15 * created on: 2004sep03
16 * created by: Markus W. Scherer
18 * Implementation class for lower-/upper-/title-casing transliterators.
21 #include "unicode/utypes.h"
23 #if !UCONFIG_NO_TRANSLITERATION
25 #include "unicode/uchar.h"
26 #include "unicode/ustring.h"
27 #include "unicode/utf.h"
28 #include "unicode/utf16.h"
33 /* case context iterator using a Replaceable */
34 U_CFUNC UChar32 U_CALLCONV
35 utrans_rep_caseContextIterator(void *context
, int8_t dir
)
39 UCaseContext
*csc
=(UCaseContext
*)context
;
40 Replaceable
*rep
=(Replaceable
*)csc
->p
;
44 /* reset for backward iteration */
45 csc
->index
=csc
->cpStart
;
48 /* reset for forward iteration */
49 csc
->index
=csc
->cpLimit
;
52 /* continue current iteration direction */
56 // automatically adjust start and limit if the Replaceable disagrees
57 // with the original values
59 if(csc
->start
<csc
->index
) {
60 c
=rep
->char32At(csc
->index
-1);
62 csc
->start
=csc
->index
;
64 csc
->index
-=U16_LENGTH(c
);
69 // detect, and store in csc->b1, if we hit the limit
70 if(csc
->index
<csc
->limit
) {
71 c
=rep
->char32At(csc
->index
);
73 csc
->limit
=csc
->index
;
76 csc
->index
+=U16_LENGTH(c
);
88 UOBJECT_DEFINE_ABSTRACT_RTTI_IMPLEMENTATION(CaseMapTransliterator
)
91 * Constructs a transliterator.
93 CaseMapTransliterator::CaseMapTransliterator(const UnicodeString
&id
, UCaseMapFull
*map
) :
94 Transliterator(id
, 0),
97 // TODO test incremental mode with context-sensitive text (e.g. greek sigma)
98 // TODO need to call setMaximumContextLength()?!
104 CaseMapTransliterator::~CaseMapTransliterator() {
110 CaseMapTransliterator::CaseMapTransliterator(const CaseMapTransliterator
& o
) :
117 * Assignment operator.
119 /*CaseMapTransliterator& CaseMapTransliterator::operator=(const CaseMapTransliterator& o) {
120 Transliterator::operator=(o);
126 * Transliterator API.
128 /*CaseMapTransliterator* CaseMapTransliterator::clone() const {
129 return new CaseMapTransliterator(*this);
133 * Implements {@link Transliterator#handleTransliterate}.
135 void CaseMapTransliterator::handleTransliterate(Replaceable
& text
,
136 UTransPosition
& offsets
,
137 UBool isIncremental
) const
139 if (offsets
.start
>= offsets
.limit
) {
144 uprv_memset(&csc
, 0, sizeof(csc
));
146 csc
.start
= offsets
.contextStart
;
147 csc
.limit
= offsets
.contextLimit
;
152 int32_t textPos
, delta
, result
;
154 for(textPos
=offsets
.start
; textPos
<offsets
.limit
;) {
156 c
=text
.char32At(textPos
);
157 csc
.cpLimit
=textPos
+=U16_LENGTH(c
);
159 result
=fMap(c
, utrans_rep_caseContextIterator
, &csc
, &s
, UCASE_LOC_ROOT
);
161 if(csc
.b1
&& isIncremental
) {
162 // fMap() tried to look beyond the context limit
163 // wait for more input
164 offsets
.start
=csc
.cpStart
;
169 // replace the current code point with its full case mapping result
170 // see UCASE_MAX_STRING_LENGTH
171 if(result
<=UCASE_MAX_STRING_LENGTH
) {
173 tmp
.setTo(FALSE
, s
, result
);
174 delta
=result
-U16_LENGTH(c
);
178 delta
=tmp
.length()-U16_LENGTH(c
);
180 text
.handleReplaceBetween(csc
.cpStart
, textPos
, tmp
);
183 csc
.limit
=offsets
.contextLimit
+=delta
;
184 offsets
.limit
+=delta
;
188 offsets
.start
=textPos
;
193 #endif /* #if !UCONFIG_NO_TRANSLITERATION */