1 /*
2 * Copyright (C) 2015 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17 #include <gtest/gtest.h>
18 #include <UnicodeUtils.h>
19
20 #include "LayoutUtils.h"
21
22 namespace minikin {
23
ExpectNextWordBreakForCache(size_t offset_in,const char * query_str)24 void ExpectNextWordBreakForCache(size_t offset_in, const char* query_str) {
25 const size_t BUF_SIZE = 256U;
26 uint16_t buf[BUF_SIZE];
27 size_t expected_breakpoint = 0U;
28 size_t size = 0U;
29
30 ParseUnicode(buf, BUF_SIZE, query_str, &size, &expected_breakpoint);
31 EXPECT_EQ(expected_breakpoint,
32 getNextWordBreakForCache(buf, offset_in, size))
33 << "Expected position is [" << query_str << "] from offset " << offset_in;
34 }
35
ExpectPrevWordBreakForCache(size_t offset_in,const char * query_str)36 void ExpectPrevWordBreakForCache(size_t offset_in, const char* query_str) {
37 const size_t BUF_SIZE = 256U;
38 uint16_t buf[BUF_SIZE];
39 size_t expected_breakpoint = 0U;
40 size_t size = 0U;
41
42 ParseUnicode(buf, BUF_SIZE, query_str, &size, &expected_breakpoint);
43 EXPECT_EQ(expected_breakpoint,
44 getPrevWordBreakForCache(buf, offset_in, size))
45 << "Expected position is [" << query_str << "] from offset " << offset_in;
46 }
47
TEST(WordBreakTest,goNextWordBreakTest)48 TEST(WordBreakTest, goNextWordBreakTest) {
49 ExpectNextWordBreakForCache(0, "|");
50
51 // Continue for spaces.
52 ExpectNextWordBreakForCache(0, "'a' 'b' 'c' 'd' |");
53 ExpectNextWordBreakForCache(1, "'a' 'b' 'c' 'd' |");
54 ExpectNextWordBreakForCache(2, "'a' 'b' 'c' 'd' |");
55 ExpectNextWordBreakForCache(3, "'a' 'b' 'c' 'd' |");
56 ExpectNextWordBreakForCache(4, "'a' 'b' 'c' 'd' |");
57 ExpectNextWordBreakForCache(1000, "'a' 'b' 'c' 'd' |");
58
59 // Space makes word break.
60 ExpectNextWordBreakForCache(0, "'a' 'b' | U+0020 'c' 'd'");
61 ExpectNextWordBreakForCache(1, "'a' 'b' | U+0020 'c' 'd'");
62 ExpectNextWordBreakForCache(2, "'a' 'b' U+0020 | 'c' 'd'");
63 ExpectNextWordBreakForCache(3, "'a' 'b' U+0020 'c' 'd' |");
64 ExpectNextWordBreakForCache(4, "'a' 'b' U+0020 'c' 'd' |");
65 ExpectNextWordBreakForCache(5, "'a' 'b' U+0020 'c' 'd' |");
66 ExpectNextWordBreakForCache(1000, "'a' 'b' U+0020 'c' 'd' |");
67
68 ExpectNextWordBreakForCache(0, "'a' 'b' | U+2000 'c' 'd'");
69 ExpectNextWordBreakForCache(1, "'a' 'b' | U+2000 'c' 'd'");
70 ExpectNextWordBreakForCache(2, "'a' 'b' U+2000 | 'c' 'd'");
71 ExpectNextWordBreakForCache(3, "'a' 'b' U+2000 'c' 'd' |");
72 ExpectNextWordBreakForCache(4, "'a' 'b' U+2000 'c' 'd' |");
73 ExpectNextWordBreakForCache(5, "'a' 'b' U+2000 'c' 'd' |");
74 ExpectNextWordBreakForCache(1000, "'a' 'b' U+2000 'c' 'd' |");
75
76 ExpectNextWordBreakForCache(0, "'a' 'b' | U+2000 U+2000 'c' 'd'");
77 ExpectNextWordBreakForCache(1, "'a' 'b' | U+2000 U+2000 'c' 'd'");
78 ExpectNextWordBreakForCache(2, "'a' 'b' U+2000 | U+2000 'c' 'd'");
79 ExpectNextWordBreakForCache(3, "'a' 'b' U+2000 U+2000 | 'c' 'd'");
80 ExpectNextWordBreakForCache(4, "'a' 'b' U+2000 U+2000 'c' 'd' |");
81 ExpectNextWordBreakForCache(5, "'a' 'b' U+2000 U+2000 'c' 'd' |");
82 ExpectNextWordBreakForCache(6, "'a' 'b' U+2000 U+2000 'c' 'd' |");
83 ExpectNextWordBreakForCache(1000, "'a' 'b' U+2000 U+2000 'c' 'd' |");
84
85 // CJK ideographs makes word break.
86 ExpectNextWordBreakForCache(0, "U+4E00 | U+4E00 U+4E00 U+4E00 U+4E00");
87 ExpectNextWordBreakForCache(1, "U+4E00 U+4E00 | U+4E00 U+4E00 U+4E00");
88 ExpectNextWordBreakForCache(2, "U+4E00 U+4E00 U+4E00 | U+4E00 U+4E00");
89 ExpectNextWordBreakForCache(3, "U+4E00 U+4E00 U+4E00 U+4E00 | U+4E00");
90 ExpectNextWordBreakForCache(4, "U+4E00 U+4E00 U+4E00 U+4E00 U+4E00 |");
91 ExpectNextWordBreakForCache(5, "U+4E00 U+4E00 U+4E00 U+4E00 U+4E00 |");
92 ExpectNextWordBreakForCache(1000,
93 "U+4E00 U+4E00 U+4E00 U+4E00 U+4E00 |");
94
95 ExpectNextWordBreakForCache(0, "U+4E00 | U+4E8C U+4E09 U+56DB U+4E94");
96 ExpectNextWordBreakForCache(1, "U+4E00 U+4E8C | U+4E09 U+56DB U+4E94");
97 ExpectNextWordBreakForCache(2, "U+4E00 U+4E8C U+4E09 | U+56DB U+4E94");
98 ExpectNextWordBreakForCache(3, "U+4E00 U+4E8C U+4E09 U+56DB | U+4E94");
99 ExpectNextWordBreakForCache(4, "U+4E00 U+4E8C U+4E09 U+56DB U+4E94 |");
100 ExpectNextWordBreakForCache(5, "U+4E00 U+4E8C U+4E09 U+56DB U+4E94 |");
101 ExpectNextWordBreakForCache(1000,
102 "U+4E00 U+4E8C U+4E09 U+56DB U+4E94 |");
103
104 ExpectNextWordBreakForCache(0, "U+4E00 'a' 'b' | U+2000 'c' U+4E00");
105 ExpectNextWordBreakForCache(1, "U+4E00 'a' 'b' | U+2000 'c' U+4E00");
106 ExpectNextWordBreakForCache(2, "U+4E00 'a' 'b' | U+2000 'c' U+4E00");
107 ExpectNextWordBreakForCache(3, "U+4E00 'a' 'b' U+2000 | 'c' U+4E00");
108 ExpectNextWordBreakForCache(4, "U+4E00 'a' 'b' U+2000 'c' | U+4E00");
109 ExpectNextWordBreakForCache(5, "U+4E00 'a' 'b' U+2000 'c' U+4E00 |");
110 ExpectNextWordBreakForCache(1000, "U+4E00 'a' 'b' U+2000 'c' U+4E00 |");
111
112 // Continue if trailing characters is Unicode combining characters.
113 ExpectNextWordBreakForCache(0, "U+4E00 U+0332 | U+4E00");
114 ExpectNextWordBreakForCache(1, "U+4E00 U+0332 | U+4E00");
115 ExpectNextWordBreakForCache(2, "U+4E00 U+0332 U+4E00 |");
116 ExpectNextWordBreakForCache(3, "U+4E00 U+0332 U+4E00 |");
117 ExpectNextWordBreakForCache(1000, "U+4E00 U+0332 U+4E00 |");
118
119 // Surrogate pairs.
120 ExpectNextWordBreakForCache(0, "U+1F60D U+1F618 |");
121 ExpectNextWordBreakForCache(1, "U+1F60D U+1F618 |");
122 ExpectNextWordBreakForCache(2, "U+1F60D U+1F618 |");
123 ExpectNextWordBreakForCache(3, "U+1F60D U+1F618 |");
124 ExpectNextWordBreakForCache(4, "U+1F60D U+1F618 |");
125 ExpectNextWordBreakForCache(1000, "U+1F60D U+1F618 |");
126
127 // Broken surrogate pairs.
128 // U+D84D is leading surrogate but there is no trailing surrogate for it.
129 ExpectNextWordBreakForCache(0, "U+D84D U+1F618 |");
130 ExpectNextWordBreakForCache(1, "U+D84D U+1F618 |");
131 ExpectNextWordBreakForCache(2, "U+D84D U+1F618 |");
132 ExpectNextWordBreakForCache(3, "U+D84D U+1F618 |");
133 ExpectNextWordBreakForCache(1000, "U+D84D U+1F618 |");
134
135 ExpectNextWordBreakForCache(0, "U+1F618 U+D84D |");
136 ExpectNextWordBreakForCache(1, "U+1F618 U+D84D |");
137 ExpectNextWordBreakForCache(2, "U+1F618 U+D84D |");
138 ExpectNextWordBreakForCache(3, "U+1F618 U+D84D |");
139 ExpectNextWordBreakForCache(1000, "U+1F618 U+D84D |");
140
141 // U+DE0D is trailing surrogate but there is no leading surrogate for it.
142 ExpectNextWordBreakForCache(0, "U+DE0D U+1F618 |");
143 ExpectNextWordBreakForCache(1, "U+DE0D U+1F618 |");
144 ExpectNextWordBreakForCache(2, "U+DE0D U+1F618 |");
145 ExpectNextWordBreakForCache(3, "U+DE0D U+1F618 |");
146 ExpectNextWordBreakForCache(1000, "U+DE0D U+1F618 |");
147
148 ExpectNextWordBreakForCache(0, "U+1F618 U+DE0D |");
149 ExpectNextWordBreakForCache(1, "U+1F618 U+DE0D |");
150 ExpectNextWordBreakForCache(2, "U+1F618 U+DE0D |");
151 ExpectNextWordBreakForCache(3, "U+1F618 U+DE0D |");
152 ExpectNextWordBreakForCache(1000, "U+1F618 U+DE0D |");
153
154 // Regional indicator pair. U+1F1FA U+1F1F8 is US national flag.
155 ExpectNextWordBreakForCache(0, "U+1F1FA U+1F1F8 |");
156 ExpectNextWordBreakForCache(1, "U+1F1FA U+1F1F8 |");
157 ExpectNextWordBreakForCache(2, "U+1F1FA U+1F1F8 |");
158 ExpectNextWordBreakForCache(1000, "U+1F1FA U+1F1F8 |");
159
160 // Tone marks.
161 // CJK ideographic char + Tone mark + CJK ideographic char
162 ExpectNextWordBreakForCache(0, "U+4444 U+302D | U+4444");
163 ExpectNextWordBreakForCache(1, "U+4444 U+302D | U+4444");
164 ExpectNextWordBreakForCache(2, "U+4444 U+302D U+4444 |");
165 ExpectNextWordBreakForCache(3, "U+4444 U+302D U+4444 |");
166 ExpectNextWordBreakForCache(1000, "U+4444 U+302D U+4444 |");
167
168 // Variation Selectors.
169 // CJK Ideographic char + Variation Selector(VS1) + CJK Ideographic char
170 ExpectNextWordBreakForCache(0, "U+845B U+FE00 | U+845B");
171 ExpectNextWordBreakForCache(1, "U+845B U+FE00 | U+845B");
172 ExpectNextWordBreakForCache(2, "U+845B U+FE00 U+845B |");
173 ExpectNextWordBreakForCache(3, "U+845B U+FE00 U+845B |");
174 ExpectNextWordBreakForCache(1000, "U+845B U+FE00 U+845B |");
175
176 // CJK Ideographic char + Variation Selector(VS17) + CJK Ideographic char
177 ExpectNextWordBreakForCache(0, "U+845B U+E0100 | U+845B");
178 ExpectNextWordBreakForCache(1, "U+845B U+E0100 | U+845B");
179 ExpectNextWordBreakForCache(2, "U+845B U+E0100 | U+845B");
180 ExpectNextWordBreakForCache(3, "U+845B U+E0100 U+845B |");
181 ExpectNextWordBreakForCache(4, "U+845B U+E0100 U+845B |");
182 ExpectNextWordBreakForCache(5, "U+845B U+E0100 U+845B |");
183 ExpectNextWordBreakForCache(1000, "U+845B U+E0100 U+845B |");
184
185 // CJK ideographic char + Tone mark + Variation Character(VS1)
186 ExpectNextWordBreakForCache(0, "U+4444 U+302D U+FE00 | U+4444");
187 ExpectNextWordBreakForCache(1, "U+4444 U+302D U+FE00 | U+4444");
188 ExpectNextWordBreakForCache(2, "U+4444 U+302D U+FE00 | U+4444");
189 ExpectNextWordBreakForCache(3, "U+4444 U+302D U+FE00 U+4444 |");
190 ExpectNextWordBreakForCache(4, "U+4444 U+302D U+FE00 U+4444 |");
191 ExpectNextWordBreakForCache(1000, "U+4444 U+302D U+FE00 U+4444 |");
192
193 // CJK ideographic char + Tone mark + Variation Character(VS17)
194 ExpectNextWordBreakForCache(0, "U+4444 U+302D U+E0100 | U+4444");
195 ExpectNextWordBreakForCache(1, "U+4444 U+302D U+E0100 | U+4444");
196 ExpectNextWordBreakForCache(2, "U+4444 U+302D U+E0100 | U+4444");
197 ExpectNextWordBreakForCache(3, "U+4444 U+302D U+E0100 | U+4444");
198 ExpectNextWordBreakForCache(4, "U+4444 U+302D U+E0100 U+4444 |");
199 ExpectNextWordBreakForCache(5, "U+4444 U+302D U+E0100 U+4444 |");
200 ExpectNextWordBreakForCache(1000, "U+4444 U+302D U+E0100 U+4444 |");
201
202 // CJK ideographic char + Variation Character(VS1) + Tone mark
203 ExpectNextWordBreakForCache(0, "U+4444 U+FE00 U+302D | U+4444");
204 ExpectNextWordBreakForCache(1, "U+4444 U+FE00 U+302D | U+4444");
205 ExpectNextWordBreakForCache(2, "U+4444 U+FE00 U+302D | U+4444");
206 ExpectNextWordBreakForCache(3, "U+4444 U+FE00 U+302D U+4444 |");
207 ExpectNextWordBreakForCache(4, "U+4444 U+FE00 U+302D U+4444 |");
208 ExpectNextWordBreakForCache(1000, "U+4444 U+FE00 U+302D U+4444 |");
209
210 // CJK ideographic char + Variation Character(VS17) + Tone mark
211 ExpectNextWordBreakForCache(0, "U+4444 U+E0100 U+302D | U+4444");
212 ExpectNextWordBreakForCache(1, "U+4444 U+E0100 U+302D | U+4444");
213 ExpectNextWordBreakForCache(2, "U+4444 U+E0100 U+302D | U+4444");
214 ExpectNextWordBreakForCache(3, "U+4444 U+E0100 U+302D | U+4444");
215 ExpectNextWordBreakForCache(4, "U+4444 U+E0100 U+302D U+4444 |");
216 ExpectNextWordBreakForCache(5, "U+4444 U+E0100 U+302D U+4444 |");
217 ExpectNextWordBreakForCache(1000, "U+4444 U+E0100 U+302D U+4444 |");
218
219 // Following test cases are unusual usage of variation selectors and tone
220 // marks for caching up the further behavior changes, e.g. index of bounds
221 // or crashes. Please feel free to update the test expectations if the
222 // behavior change makes sense to you.
223
224 // Isolated Tone marks and Variation Selectors
225 ExpectNextWordBreakForCache(0, "U+FE00 |");
226 ExpectNextWordBreakForCache(1, "U+FE00 |");
227 ExpectNextWordBreakForCache(1000, "U+FE00 |");
228 ExpectNextWordBreakForCache(0, "U+E0100 |");
229 ExpectNextWordBreakForCache(1000, "U+E0100 |");
230 ExpectNextWordBreakForCache(0, "U+302D |");
231 ExpectNextWordBreakForCache(1000, "U+302D |");
232
233 // CJK Ideographic char + Variation Selector(VS1) + Variation Selector(VS1)
234 ExpectNextWordBreakForCache(0, "U+845B U+FE00 U+FE00 | U+845B");
235 ExpectNextWordBreakForCache(1, "U+845B U+FE00 U+FE00 | U+845B");
236 ExpectNextWordBreakForCache(2, "U+845B U+FE00 U+FE00 | U+845B");
237 ExpectNextWordBreakForCache(3, "U+845B U+FE00 U+FE00 U+845B |");
238 ExpectNextWordBreakForCache(4, "U+845B U+FE00 U+FE00 U+845B |");
239 ExpectNextWordBreakForCache(1000, "U+845B U+FE00 U+FE00 U+845B |");
240
241 // CJK Ideographic char + Variation Selector(VS17) + Variation Selector(VS17)
242 ExpectNextWordBreakForCache(0, "U+845B U+E0100 U+E0100 | U+845B");
243 ExpectNextWordBreakForCache(1, "U+845B U+E0100 U+E0100 | U+845B");
244 ExpectNextWordBreakForCache(2, "U+845B U+E0100 U+E0100 | U+845B");
245 ExpectNextWordBreakForCache(3, "U+845B U+E0100 U+E0100 | U+845B");
246 ExpectNextWordBreakForCache(4, "U+845B U+E0100 U+E0100 | U+845B");
247 ExpectNextWordBreakForCache(5, "U+845B U+E0100 U+E0100 U+845B |");
248 ExpectNextWordBreakForCache(6, "U+845B U+E0100 U+E0100 U+845B |");
249 ExpectNextWordBreakForCache(1000,
250 "U+845B U+E0100 U+E0100 U+845B |");
251
252 // CJK Ideographic char + Variation Selector(VS1) + Variation Selector(VS17)
253 ExpectNextWordBreakForCache(0, "U+845B U+FE00 U+E0100 | U+845B");
254 ExpectNextWordBreakForCache(1, "U+845B U+FE00 U+E0100 | U+845B");
255 ExpectNextWordBreakForCache(2, "U+845B U+FE00 U+E0100 | U+845B");
256 ExpectNextWordBreakForCache(3, "U+845B U+FE00 U+E0100 | U+845B");
257 ExpectNextWordBreakForCache(4, "U+845B U+FE00 U+E0100 U+845B |");
258 ExpectNextWordBreakForCache(5, "U+845B U+FE00 U+E0100 U+845B |");
259 ExpectNextWordBreakForCache(1000, "U+845B U+FE00 U+E0100 U+845B |");
260
261 // CJK Ideographic char + Variation Selector(VS17) + Variation Selector(VS1)
262 ExpectNextWordBreakForCache(0, "U+845B U+E0100 U+FE00 | U+845B");
263 ExpectNextWordBreakForCache(1, "U+845B U+E0100 U+FE00 | U+845B");
264 ExpectNextWordBreakForCache(2, "U+845B U+E0100 U+FE00 | U+845B");
265 ExpectNextWordBreakForCache(3, "U+845B U+E0100 U+FE00 | U+845B");
266 ExpectNextWordBreakForCache(4, "U+845B U+E0100 U+FE00 U+845B |");
267 ExpectNextWordBreakForCache(5, "U+845B U+E0100 U+FE00 U+845B |");
268 ExpectNextWordBreakForCache(1000, "U+845B U+E0100 U+FE00 U+845B |");
269
270 // Tone mark. + Tone mark
271 ExpectNextWordBreakForCache(0, "U+4444 U+302D U+302D | U+4444");
272 ExpectNextWordBreakForCache(1, "U+4444 U+302D U+302D | U+4444");
273 ExpectNextWordBreakForCache(2, "U+4444 U+302D U+302D | U+4444");
274 ExpectNextWordBreakForCache(3, "U+4444 U+302D U+302D U+4444 |");
275 ExpectNextWordBreakForCache(4, "U+4444 U+302D U+302D U+4444 |");
276 ExpectNextWordBreakForCache(1000, "U+4444 U+302D U+302D U+4444 |");
277 }
278
TEST(WordBreakTest,goPrevWordBreakTest)279 TEST(WordBreakTest, goPrevWordBreakTest) {
280 ExpectPrevWordBreakForCache(0, "|");
281
282 // Continue for spaces.
283 ExpectPrevWordBreakForCache(0, "| 'a' 'b' 'c' 'd'");
284 ExpectPrevWordBreakForCache(1, "| 'a' 'b' 'c' 'd'");
285 ExpectPrevWordBreakForCache(2, "| 'a' 'b' 'c' 'd'");
286 ExpectPrevWordBreakForCache(3, "| 'a' 'b' 'c' 'd'");
287 ExpectPrevWordBreakForCache(4, "| 'a' 'b' 'c' 'd'");
288 ExpectPrevWordBreakForCache(1000, "| 'a' 'b' 'c' 'd'");
289
290 // Space makes word break.
291 ExpectPrevWordBreakForCache(0, "| 'a' 'b' U+0020 'c' 'd'");
292 ExpectPrevWordBreakForCache(1, "| 'a' 'b' U+0020 'c' 'd'");
293 ExpectPrevWordBreakForCache(2, "| 'a' 'b' U+0020 'c' 'd'");
294 ExpectPrevWordBreakForCache(3, "'a' 'b' | U+0020 'c' 'd'");
295 ExpectPrevWordBreakForCache(4, "'a' 'b' U+0020 | 'c' 'd'");
296 ExpectPrevWordBreakForCache(5, "'a' 'b' U+0020 | 'c' 'd'");
297 ExpectPrevWordBreakForCache(1000, "'a' 'b' U+0020 | 'c' 'd'");
298
299 ExpectPrevWordBreakForCache(0, "| 'a' 'b' U+2000 'c' 'd'");
300 ExpectPrevWordBreakForCache(1, "| 'a' 'b' U+2000 'c' 'd'");
301 ExpectPrevWordBreakForCache(2, "| 'a' 'b' U+2000 'c' 'd'");
302 ExpectPrevWordBreakForCache(3, "'a' 'b' | U+2000 'c' 'd'");
303 ExpectPrevWordBreakForCache(4, "'a' 'b' U+2000 | 'c' 'd'");
304 ExpectPrevWordBreakForCache(5, "'a' 'b' U+2000 | 'c' 'd'");
305 ExpectPrevWordBreakForCache(1000, "'a' 'b' U+2000 | 'c' 'd'");
306
307 ExpectPrevWordBreakForCache(0, "| 'a' 'b' U+2000 U+2000 'c' 'd'");
308 ExpectPrevWordBreakForCache(1, "| 'a' 'b' U+2000 U+2000 'c' 'd'");
309 ExpectPrevWordBreakForCache(2, "| 'a' 'b' U+2000 U+2000 'c' 'd'");
310 ExpectPrevWordBreakForCache(3, "'a' 'b' | U+2000 U+2000 'c' 'd'");
311 ExpectPrevWordBreakForCache(4, "'a' 'b' U+2000 | U+2000 'c' 'd'");
312 ExpectPrevWordBreakForCache(5, "'a' 'b' U+2000 U+2000 | 'c' 'd'");
313 ExpectPrevWordBreakForCache(6, "'a' 'b' U+2000 U+2000 | 'c' 'd'");
314 ExpectPrevWordBreakForCache(1000, "'a' 'b' U+2000 U+2000 | 'c' 'd'");
315
316 // CJK ideographs makes word break.
317 ExpectPrevWordBreakForCache(0, "| U+4E00 U+4E00 U+4E00 U+4E00 U+4E00");
318 ExpectPrevWordBreakForCache(1, "| U+4E00 U+4E00 U+4E00 U+4E00 U+4E00");
319 ExpectPrevWordBreakForCache(2, "U+4E00 | U+4E00 U+4E00 U+4E00 U+4E00");
320 ExpectPrevWordBreakForCache(3, "U+4E00 U+4E00 | U+4E00 U+4E00 U+4E00");
321 ExpectPrevWordBreakForCache(4, "U+4E00 U+4E00 U+4E00 | U+4E00 U+4E00");
322 ExpectPrevWordBreakForCache(5, "U+4E00 U+4E00 U+4E00 U+4E00 | U+4E00");
323 ExpectPrevWordBreakForCache(1000, "U+4E00 U+4E00 U+4E00 U+4E00 | U+4E00");
324
325 ExpectPrevWordBreakForCache(0, "| U+4E00 U+4E8C U+4E09 U+56DB U+4E94");
326 ExpectPrevWordBreakForCache(1, "| U+4E00 U+4E8C U+4E09 U+56DB U+4E94");
327 ExpectPrevWordBreakForCache(2, "U+4E00 | U+4E8C U+4E09 U+56DB U+4E94");
328 ExpectPrevWordBreakForCache(3, "U+4E00 U+4E8C | U+4E09 U+56DB U+4E94");
329 ExpectPrevWordBreakForCache(4, "U+4E00 U+4E8C U+4E09 | U+56DB U+4E94");
330 ExpectPrevWordBreakForCache(5, "U+4E00 U+4E8C U+4E09 U+56DB | U+4E94");
331 ExpectPrevWordBreakForCache(1000, "U+4E00 U+4E8C U+4E09 U+56DB | U+4E94");
332
333 // Mixed case.
334 ExpectPrevWordBreakForCache(0, "| U+4E00 'a' 'b' U+2000 'c' U+4E00");
335 ExpectPrevWordBreakForCache(1, "| U+4E00 'a' 'b' U+2000 'c' U+4E00");
336 ExpectPrevWordBreakForCache(2, "| U+4E00 'a' 'b' U+2000 'c' U+4E00");
337 ExpectPrevWordBreakForCache(3, "| U+4E00 'a' 'b' U+2000 'c' U+4E00");
338 ExpectPrevWordBreakForCache(4, "U+4E00 'a' 'b' | U+2000 'c' U+4E00");
339 ExpectPrevWordBreakForCache(5, "U+4E00 'a' 'b' U+2000 | 'c' U+4E00");
340 ExpectPrevWordBreakForCache(6, "U+4E00 'a' 'b' U+2000 'c' | U+4E00");
341 ExpectPrevWordBreakForCache(1000, "U+4E00 'a' 'b' U+2000 'c' | U+4E00");
342
343 // Continue if trailing characters is Unicode combining characters.
344 ExpectPrevWordBreakForCache(0, "| U+4E00 U+0332 U+4E00");
345 ExpectPrevWordBreakForCache(1, "| U+4E00 U+0332 U+4E00");
346 ExpectPrevWordBreakForCache(2, "| U+4E00 U+0332 U+4E00");
347 ExpectPrevWordBreakForCache(3, "U+4E00 U+0332 | U+4E00");
348 ExpectPrevWordBreakForCache(1000, "U+4E00 U+0332 | U+4E00");
349
350 // Surrogate pairs.
351 ExpectPrevWordBreakForCache(0, "| U+1F60D U+1F618");
352 ExpectPrevWordBreakForCache(1, "| U+1F60D U+1F618");
353 ExpectPrevWordBreakForCache(2, "| U+1F60D U+1F618");
354 ExpectPrevWordBreakForCache(3, "| U+1F60D U+1F618");
355 ExpectPrevWordBreakForCache(4, "| U+1F60D U+1F618");
356 ExpectPrevWordBreakForCache(1000, "| U+1F60D U+1F618");
357
358 // Broken surrogate pairs.
359 // U+D84D is leading surrogate but there is no trailing surrogate for it.
360 ExpectPrevWordBreakForCache(0, "| U+D84D U+1F618");
361 ExpectPrevWordBreakForCache(1, "| U+D84D U+1F618");
362 ExpectPrevWordBreakForCache(2, "| U+D84D U+1F618");
363 ExpectPrevWordBreakForCache(3, "| U+D84D U+1F618");
364 ExpectPrevWordBreakForCache(1000, "| U+D84D U+1F618");
365
366 ExpectPrevWordBreakForCache(0, "| U+1F618 U+D84D");
367 ExpectPrevWordBreakForCache(1, "| U+1F618 U+D84D");
368 ExpectPrevWordBreakForCache(2, "| U+1F618 U+D84D");
369 ExpectPrevWordBreakForCache(3, "| U+1F618 U+D84D");
370 ExpectPrevWordBreakForCache(1000, "| U+1F618 U+D84D");
371
372 // U+DE0D is trailing surrogate but there is no leading surrogate for it.
373 ExpectPrevWordBreakForCache(0, "| U+DE0D U+1F618");
374 ExpectPrevWordBreakForCache(1, "| U+DE0D U+1F618");
375 ExpectPrevWordBreakForCache(2, "| U+DE0D U+1F618");
376 ExpectPrevWordBreakForCache(3, "| U+DE0D U+1F618");
377 ExpectPrevWordBreakForCache(1000, "| U+DE0D U+1F618");
378
379 ExpectPrevWordBreakForCache(0, "| U+1F618 U+DE0D");
380 ExpectPrevWordBreakForCache(1, "| U+1F618 U+DE0D");
381 ExpectPrevWordBreakForCache(2, "| U+1F618 U+DE0D");
382 ExpectPrevWordBreakForCache(3, "| U+1F618 U+DE0D");
383 ExpectPrevWordBreakForCache(1000, "| U+1F618 U+DE0D");
384
385 // Regional indicator pair. U+1F1FA U+1F1F8 is US national flag.
386 ExpectPrevWordBreakForCache(0, "| U+1F1FA U+1F1F8");
387 ExpectPrevWordBreakForCache(1, "| U+1F1FA U+1F1F8");
388 ExpectPrevWordBreakForCache(2, "| U+1F1FA U+1F1F8");
389 ExpectPrevWordBreakForCache(1000, "| U+1F1FA U+1F1F8");
390
391 // Tone marks.
392 // CJK ideographic char + Tone mark + CJK ideographic char
393 ExpectPrevWordBreakForCache(0, "| U+4444 U+302D U+4444");
394 ExpectPrevWordBreakForCache(1, "| U+4444 U+302D U+4444");
395 ExpectPrevWordBreakForCache(2, "| U+4444 U+302D U+4444");
396 ExpectPrevWordBreakForCache(3, "U+4444 U+302D | U+4444");
397 ExpectPrevWordBreakForCache(1000, "U+4444 U+302D | U+4444");
398
399 // Variation Selectors.
400 // CJK Ideographic char + Variation Selector(VS1) + CJK Ideographic char
401 ExpectPrevWordBreakForCache(0, "| U+845B U+FE00 U+845B");
402 ExpectPrevWordBreakForCache(1, "| U+845B U+FE00 U+845B");
403 ExpectPrevWordBreakForCache(2, "| U+845B U+FE00 U+845B");
404 ExpectPrevWordBreakForCache(3, "U+845B U+FE00 | U+845B");
405 ExpectPrevWordBreakForCache(1000, "U+845B U+FE00 | U+845B");
406
407 // CJK Ideographic char + Variation Selector(VS17) + CJK Ideographic char
408 ExpectPrevWordBreakForCache(0, "| U+845B U+E0100 U+845B");
409 ExpectPrevWordBreakForCache(1, "| U+845B U+E0100 U+845B");
410 ExpectPrevWordBreakForCache(2, "| U+845B U+E0100 U+845B");
411 ExpectPrevWordBreakForCache(3, "| U+845B U+E0100 U+845B");
412 ExpectPrevWordBreakForCache(4, "U+845B U+E0100 | U+845B");
413 ExpectPrevWordBreakForCache(5, "U+845B U+E0100 | U+845B");
414 ExpectPrevWordBreakForCache(1000, "U+845B U+E0100 | U+845B");
415
416 // CJK ideographic char + Tone mark + Variation Character(VS1)
417 ExpectPrevWordBreakForCache(0, "| U+4444 U+302D U+FE00 U+4444");
418 ExpectPrevWordBreakForCache(1, "| U+4444 U+302D U+FE00 U+4444");
419 ExpectPrevWordBreakForCache(2, "| U+4444 U+302D U+FE00 U+4444");
420 ExpectPrevWordBreakForCache(3, "| U+4444 U+302D U+FE00 U+4444");
421 ExpectPrevWordBreakForCache(4, "U+4444 U+302D U+FE00 | U+4444");
422 ExpectPrevWordBreakForCache(1000, "U+4444 U+302D U+FE00 | U+4444");
423
424 // CJK ideographic char + Tone mark + Variation Character(VS17)
425 ExpectPrevWordBreakForCache(0, "| U+4444 U+302D U+E0100 U+4444");
426 ExpectPrevWordBreakForCache(1, "| U+4444 U+302D U+E0100 U+4444");
427 ExpectPrevWordBreakForCache(2, "| U+4444 U+302D U+E0100 U+4444");
428 ExpectPrevWordBreakForCache(3, "| U+4444 U+302D U+E0100 U+4444");
429 ExpectPrevWordBreakForCache(4, "| U+4444 U+302D U+E0100 U+4444");
430 ExpectPrevWordBreakForCache(5, "U+4444 U+302D U+E0100 | U+4444");
431 ExpectPrevWordBreakForCache(1000, "U+4444 U+302D U+E0100 | U+4444");
432
433 // CJK ideographic char + Variation Character(VS1) + Tone mark
434 ExpectPrevWordBreakForCache(0, "| U+4444 U+FE00 U+302D U+4444");
435 ExpectPrevWordBreakForCache(1, "| U+4444 U+FE00 U+302D U+4444");
436 ExpectPrevWordBreakForCache(2, "| U+4444 U+FE00 U+302D U+4444");
437 ExpectPrevWordBreakForCache(3, "| U+4444 U+FE00 U+302D U+4444");
438 ExpectPrevWordBreakForCache(4, "U+4444 U+FE00 U+302D | U+4444");
439 ExpectPrevWordBreakForCache(1000, "U+4444 U+FE00 U+302D | U+4444");
440
441 // CJK ideographic char + Variation Character(VS17) + Tone mark
442 ExpectPrevWordBreakForCache(0, "| U+4444 U+E0100 U+302D U+4444");
443 ExpectPrevWordBreakForCache(1, "| U+4444 U+E0100 U+302D U+4444");
444 ExpectPrevWordBreakForCache(2, "| U+4444 U+E0100 U+302D U+4444");
445 ExpectPrevWordBreakForCache(3, "| U+4444 U+E0100 U+302D U+4444");
446 ExpectPrevWordBreakForCache(4, "| U+4444 U+E0100 U+302D U+4444");
447 ExpectPrevWordBreakForCache(5, "U+4444 U+E0100 U+302D | U+4444");
448 ExpectPrevWordBreakForCache(1000, "U+4444 U+E0100 U+302D | U+4444");
449
450 // Following test cases are unusual usage of variation selectors and tone
451 // marks for caching up the further behavior changes, e.g. index of bounds
452 // or crashes. Please feel free to update the test expectations if the
453 // behavior change makes sense to you.
454
455 // Isolated Tone marks and Variation Selectors
456 ExpectPrevWordBreakForCache(0, "| U+FE00");
457 ExpectPrevWordBreakForCache(1, "| U+FE00");
458 ExpectPrevWordBreakForCache(1000, "| U+FE00");
459 ExpectPrevWordBreakForCache(0, "| U+E0100");
460 ExpectPrevWordBreakForCache(1000, "| U+E0100");
461 ExpectPrevWordBreakForCache(0, "| U+302D");
462 ExpectPrevWordBreakForCache(1000, "| U+302D");
463
464 // CJK Ideographic char + Variation Selector(VS1) + Variation Selector(VS1)
465 ExpectPrevWordBreakForCache(0, "| U+845B U+FE00 U+FE00 U+845B");
466 ExpectPrevWordBreakForCache(1, "| U+845B U+FE00 U+FE00 U+845B");
467 ExpectPrevWordBreakForCache(2, "| U+845B U+FE00 U+FE00 U+845B");
468 ExpectPrevWordBreakForCache(3, "| U+845B U+FE00 U+FE00 U+845B");
469 ExpectPrevWordBreakForCache(4, "U+845B U+FE00 U+FE00 | U+845B");
470 ExpectPrevWordBreakForCache(1000, "U+845B U+FE00 U+FE00 | U+845B");
471
472 // CJK Ideographic char + Variation Selector(VS17) + Variation Selector(VS17)
473 ExpectPrevWordBreakForCache(0, "| U+845B U+E0100 U+E0100 U+845B");
474 ExpectPrevWordBreakForCache(1, "| U+845B U+E0100 U+E0100 U+845B");
475 ExpectPrevWordBreakForCache(2, "| U+845B U+E0100 U+E0100 U+845B");
476 ExpectPrevWordBreakForCache(3, "| U+845B U+E0100 U+E0100 U+845B");
477 ExpectPrevWordBreakForCache(4, "| U+845B U+E0100 U+E0100 U+845B");
478 ExpectPrevWordBreakForCache(5, "| U+845B U+E0100 U+E0100 U+845B");
479 ExpectPrevWordBreakForCache(6, "U+845B U+E0100 U+E0100 | U+845B");
480 ExpectPrevWordBreakForCache(1000,
481 "U+845B U+E0100 U+E0100 | U+845B");
482
483 // CJK Ideographic char + Variation Selector(VS1) + Variation Selector(VS17)
484 ExpectPrevWordBreakForCache(0, "| U+845B U+FE00 U+E0100 U+845B");
485 ExpectPrevWordBreakForCache(1, "| U+845B U+FE00 U+E0100 U+845B");
486 ExpectPrevWordBreakForCache(2, "| U+845B U+FE00 U+E0100 U+845B");
487 ExpectPrevWordBreakForCache(3, "| U+845B U+FE00 U+E0100 U+845B");
488 ExpectPrevWordBreakForCache(4, "| U+845B U+FE00 U+E0100 U+845B");
489 ExpectPrevWordBreakForCache(5, "U+845B U+FE00 U+E0100 | U+845B");
490 ExpectPrevWordBreakForCache(1000, "U+845B U+FE00 U+E0100 | U+845B");
491
492 // CJK Ideographic char + Variation Selector(VS17) + Variation Selector(VS1)
493 ExpectPrevWordBreakForCache(0, "| U+845B U+E0100 U+FE00 U+845B");
494 ExpectPrevWordBreakForCache(1, "| U+845B U+E0100 U+FE00 U+845B");
495 ExpectPrevWordBreakForCache(2, "| U+845B U+E0100 U+FE00 U+845B");
496 ExpectPrevWordBreakForCache(3, "| U+845B U+E0100 U+FE00 U+845B");
497 ExpectPrevWordBreakForCache(4, "| U+845B U+E0100 U+FE00 U+845B");
498 ExpectPrevWordBreakForCache(5, "U+845B U+E0100 U+FE00 | U+845B");
499 ExpectPrevWordBreakForCache(1000, "U+845B U+E0100 U+FE00 | U+845B");
500
501 // Tone mark. + Tone mark
502 ExpectPrevWordBreakForCache(0, "| U+4444 U+302D U+302D U+4444");
503 ExpectPrevWordBreakForCache(1, "| U+4444 U+302D U+302D U+4444");
504 ExpectPrevWordBreakForCache(2, "| U+4444 U+302D U+302D U+4444");
505 ExpectPrevWordBreakForCache(3, "| U+4444 U+302D U+302D U+4444");
506 ExpectPrevWordBreakForCache(4, "U+4444 U+302D U+302D | U+4444");
507 ExpectPrevWordBreakForCache(1000, "U+4444 U+302D U+302D | U+4444");
508 }
509
510 } // namespace minikin
511