1 // Copyright (c) 2011 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
5 #include "base/i18n/break_iterator.h"
7 #include "base/logging.h"
8 #include "third_party/icu/source/common/unicode/ubrk.h"
9 #include "third_party/icu/source/common/unicode/uchar.h"
10 #include "third_party/icu/source/common/unicode/ustring.h"
15 const size_t npos
= static_cast<size_t>(-1);
17 BreakIterator::BreakIterator(const string16
& str
, BreakType break_type
)
20 break_type_(break_type
),
25 BreakIterator::BreakIterator(const string16
& str
, const string16
& rules
)
29 break_type_(RULE_BASED
),
34 BreakIterator::~BreakIterator() {
36 ubrk_close(static_cast<UBreakIterator
*>(iter_
));
39 bool BreakIterator::Init() {
40 UErrorCode status
= U_ZERO_ERROR
;
41 UParseError parse_error
;
42 UBreakIteratorType break_type
;
43 switch (break_type_
) {
45 break_type
= UBRK_CHARACTER
;
48 break_type
= UBRK_WORD
;
52 case RULE_BASED
: // (Keep compiler happy, break_type not used in this case)
53 break_type
= UBRK_LINE
;
56 NOTREACHED() << "invalid break_type_";
59 if (break_type_
== RULE_BASED
) {
60 iter_
= ubrk_openRules(rules_
.c_str(),
61 static_cast<int32_t>(rules_
.length()),
63 static_cast<int32_t>(string_
.size()),
66 if (U_FAILURE(status
)) {
67 NOTREACHED() << "ubrk_openRules failed to parse rule string at line "
68 << parse_error
.line
<< ", offset " << parse_error
.offset
;
71 iter_
= ubrk_open(break_type
,
74 static_cast<int32_t>(string_
.size()),
76 if (U_FAILURE(status
)) {
77 NOTREACHED() << "ubrk_open failed";
81 if (U_FAILURE(status
)) {
85 // Move the iterator to the beginning of the string.
86 ubrk_first(static_cast<UBreakIterator
*>(iter_
));
90 bool BreakIterator::Advance() {
94 switch (break_type_
) {
99 pos
= ubrk_next(static_cast<UBreakIterator
*>(iter_
));
100 if (pos
== UBRK_DONE
) {
104 pos_
= static_cast<size_t>(pos
);
108 pos
= ubrk_next(static_cast<UBreakIterator
*>(iter_
));
109 if (pos
== UBRK_DONE
)
111 pos_
= static_cast<size_t>(pos
);
112 status
= ubrk_getRuleStatus(static_cast<UBreakIterator
*>(iter_
));
113 } while (status
>= UBRK_LINE_SOFT
&& status
< UBRK_LINE_SOFT_LIMIT
);
114 if (pos
== UBRK_DONE
&& prev_
== pos_
) {
120 NOTREACHED() << "invalid break_type_";
125 bool BreakIterator::SetText(const base::char16
* text
, const size_t length
) {
126 UErrorCode status
= U_ZERO_ERROR
;
127 ubrk_setText(static_cast<UBreakIterator
*>(iter_
),
128 text
, length
, &status
);
129 pos_
= 0; // implicit when ubrk_setText is done
131 if (U_FAILURE(status
)) {
132 NOTREACHED() << "ubrk_setText failed";
138 bool BreakIterator::IsWord() const {
139 int32_t status
= ubrk_getRuleStatus(static_cast<UBreakIterator
*>(iter_
));
140 if (break_type_
!= BREAK_WORD
&& break_type_
!= RULE_BASED
)
142 return status
!= UBRK_WORD_NONE
;
145 bool BreakIterator::IsEndOfWord(size_t position
) const {
146 if (break_type_
!= BREAK_WORD
&& break_type_
!= RULE_BASED
)
149 UBreakIterator
* iter
= static_cast<UBreakIterator
*>(iter_
);
150 UBool boundary
= ubrk_isBoundary(iter
, static_cast<int32_t>(position
));
151 int32_t status
= ubrk_getRuleStatus(iter
);
152 return (!!boundary
&& status
!= UBRK_WORD_NONE
);
155 bool BreakIterator::IsStartOfWord(size_t position
) const {
156 if (break_type_
!= BREAK_WORD
&& break_type_
!= RULE_BASED
)
159 UBreakIterator
* iter
= static_cast<UBreakIterator
*>(iter_
);
160 UBool boundary
= ubrk_isBoundary(iter
, static_cast<int32_t>(position
));
162 int32_t next_status
= ubrk_getRuleStatus(iter
);
163 return (!!boundary
&& next_status
!= UBRK_WORD_NONE
);
166 bool BreakIterator::IsGraphemeBoundary(size_t position
) const {
167 if (break_type_
!= BREAK_CHARACTER
)
170 UBreakIterator
* iter
= static_cast<UBreakIterator
*>(iter_
);
171 return !!ubrk_isBoundary(iter
, static_cast<int32_t>(position
));
174 string16
BreakIterator::GetString() const {
175 DCHECK(prev_
!= npos
&& pos_
!= npos
);
176 return string_
.substr(prev_
, pos_
- prev_
);