1 /*
2 *
3 * (C) Copyright IBM Corp. 1998-2008 - All Rights Reserved
4 *
5 */
6
7 #include "layout/LETypes.h"
8 #include "layout/loengine.h"
9 #include "layout/plruns.h"
10
11 #include "unicode/locid.h"
12
13 #include "layout/LayoutEngine.h"
14 #include "layout/RunArrays.h"
15
16 U_NAMESPACE_USE
17
18 U_CAPI pl_fontRuns * U_EXPORT2
pl_openFontRuns(const le_font ** fonts,const le_int32 * limits,le_int32 count)19 pl_openFontRuns(const le_font **fonts,
20 const le_int32 *limits,
21 le_int32 count)
22 {
23 return (pl_fontRuns *) new FontRuns((const LEFontInstance **) fonts, limits, count);
24 }
25
26 U_CAPI pl_fontRuns * U_EXPORT2
pl_openEmptyFontRuns(le_int32 initialCapacity)27 pl_openEmptyFontRuns(le_int32 initialCapacity)
28 {
29 return (pl_fontRuns *) new FontRuns(initialCapacity);
30 }
31
32 U_CAPI void U_EXPORT2
pl_closeFontRuns(pl_fontRuns * fontRuns)33 pl_closeFontRuns(pl_fontRuns *fontRuns)
34 {
35 FontRuns *fr = (FontRuns *) fontRuns;
36
37 delete fr;
38 }
39
40 U_CAPI le_int32 U_EXPORT2
pl_getFontRunCount(const pl_fontRuns * fontRuns)41 pl_getFontRunCount(const pl_fontRuns *fontRuns)
42 {
43 const FontRuns *fr = (const FontRuns *) fontRuns;
44
45 if (fr == NULL) {
46 return -1;
47 }
48
49 return fr->getCount();
50 }
51
52 U_CAPI void U_EXPORT2
pl_resetFontRuns(pl_fontRuns * fontRuns)53 pl_resetFontRuns(pl_fontRuns *fontRuns)
54 {
55 FontRuns *fr = (FontRuns *) fontRuns;
56
57 if (fr != NULL) {
58 fr->reset();
59 }
60 }
61
62 U_CAPI le_int32 U_EXPORT2
pl_getFontRunLastLimit(const pl_fontRuns * fontRuns)63 pl_getFontRunLastLimit(const pl_fontRuns *fontRuns)
64 {
65 const FontRuns *fr = (const FontRuns *) fontRuns;
66
67 if (fr == NULL) {
68 return -1;
69 }
70
71 return fr->getLimit();
72 }
73
74 U_CAPI le_int32 U_EXPORT2
pl_getFontRunLimit(const pl_fontRuns * fontRuns,le_int32 run)75 pl_getFontRunLimit(const pl_fontRuns *fontRuns,
76 le_int32 run)
77 {
78 const FontRuns *fr = (const FontRuns *) fontRuns;
79
80 if (fr == NULL) {
81 return -1;
82 }
83
84 return fr->getLimit(run);
85 }
86
87 U_CAPI const le_font * U_EXPORT2
pl_getFontRunFont(const pl_fontRuns * fontRuns,le_int32 run)88 pl_getFontRunFont(const pl_fontRuns *fontRuns,
89 le_int32 run)
90 {
91 const FontRuns *fr = (const FontRuns *) fontRuns;
92
93 if (fr == NULL) {
94 return NULL;
95 }
96
97 return (const le_font *) fr->getFont(run);
98 }
99
100 U_CAPI le_int32 U_EXPORT2
pl_addFontRun(pl_fontRuns * fontRuns,const le_font * font,le_int32 limit)101 pl_addFontRun(pl_fontRuns *fontRuns,
102 const le_font *font,
103 le_int32 limit)
104 {
105 FontRuns *fr = (FontRuns *) fontRuns;
106
107 if (fr == NULL) {
108 return -1;
109 }
110
111 return fr->add((const LEFontInstance *) font, limit);
112 }
113
114 U_CAPI pl_valueRuns * U_EXPORT2
pl_openValueRuns(const le_int32 * values,const le_int32 * limits,le_int32 count)115 pl_openValueRuns(const le_int32 *values,
116 const le_int32 *limits,
117 le_int32 count)
118 {
119 return (pl_valueRuns *) new ValueRuns(values, limits, count);
120 }
121
122 U_CAPI pl_valueRuns * U_EXPORT2
pl_openEmptyValueRuns(le_int32 initialCapacity)123 pl_openEmptyValueRuns(le_int32 initialCapacity)
124 {
125 return (pl_valueRuns *) new ValueRuns(initialCapacity);
126 }
127
128 U_CAPI void U_EXPORT2
pl_closeValueRuns(pl_valueRuns * valueRuns)129 pl_closeValueRuns(pl_valueRuns *valueRuns)
130 {
131 ValueRuns *vr = (ValueRuns *) valueRuns;
132
133 delete vr;
134 }
135
136 U_CAPI le_int32 U_EXPORT2
pl_getValueRunCount(const pl_valueRuns * valueRuns)137 pl_getValueRunCount(const pl_valueRuns *valueRuns)
138 {
139 const ValueRuns *vr = (const ValueRuns *) valueRuns;
140
141 if (vr == NULL) {
142 return -1;
143 }
144
145 return vr->getCount();
146 }
147
148 U_CAPI void U_EXPORT2
pl_resetValueRuns(pl_valueRuns * valueRuns)149 pl_resetValueRuns(pl_valueRuns *valueRuns)
150 {
151 ValueRuns *vr = (ValueRuns *) valueRuns;
152
153 if (vr != NULL) {
154 vr->reset();
155 }
156 }
157
158 U_CAPI le_int32 U_EXPORT2
pl_getValueRunLastLimit(const pl_valueRuns * valueRuns)159 pl_getValueRunLastLimit(const pl_valueRuns *valueRuns)
160 {
161 const ValueRuns *vr = (const ValueRuns *) valueRuns;
162
163 if (vr == NULL) {
164 return -1;
165 }
166
167 return vr->getLimit();
168 }
169
170 U_CAPI le_int32 U_EXPORT2
pl_getValueRunLimit(const pl_valueRuns * valueRuns,le_int32 run)171 pl_getValueRunLimit(const pl_valueRuns *valueRuns,
172 le_int32 run)
173 {
174 const ValueRuns *vr = (const ValueRuns *) valueRuns;
175
176 if (vr == NULL) {
177 return -1;
178 }
179
180 return vr->getLimit(run);
181 }
182
183 U_CAPI le_int32 U_EXPORT2
pl_getValueRunValue(const pl_valueRuns * valueRuns,le_int32 run)184 pl_getValueRunValue(const pl_valueRuns *valueRuns,
185 le_int32 run)
186 {
187 const ValueRuns *vr = (const ValueRuns *) valueRuns;
188
189 if (vr == NULL) {
190 return -1;
191 }
192
193 return vr->getValue(run);
194 }
195
196 U_CAPI le_int32 U_EXPORT2
pl_addValueRun(pl_valueRuns * valueRuns,le_int32 value,le_int32 limit)197 pl_addValueRun(pl_valueRuns *valueRuns,
198 le_int32 value,
199 le_int32 limit)
200 {
201 ValueRuns *vr = (ValueRuns *) valueRuns;
202
203 if (vr == NULL) {
204 return -1;
205 }
206
207 return vr->add(value, limit);
208 }
209
210 U_NAMESPACE_BEGIN
211 class ULocRuns : public LocaleRuns
212 {
213 public:
214 /**
215 * Construct a <code>LocaleRuns</code> object from pre-existing arrays of locales
216 * and limit indices.
217 *
218 * @param locales is the address of an array of locale name strings. This array,
219 * and the <code>Locale</code> objects to which it points, must remain valid until
220 * the <code>LocaleRuns</code> object is destroyed.
221 *
222 * @param limits is the address of an array of limit indices. This array must remain valid until the
223 * <code>LocaleRuns</code> object is destroyed.
224 *
225 * @param count is the number of entries in the two arrays.
226 *
227 * @draft ICU 3.8
228 */
229 ULocRuns(const char **locales, const le_int32 *limits, le_int32 count);
230
231 /**
232 * Construct an empty <code>LoIDRuns</code> object. Clients can add locale and limit
233 * indices arrays using the <code>add</code> method.
234 *
235 * @param initialCapacity is the initial size of the locale and limit indices arrays. If
236 * this value is zero, no arrays will be allocated.
237 *
238 * @see add
239 *
240 * @draft ICU 3.8
241 */
242 ULocRuns(le_int32 initialCapacity);
243
244 /**
245 * The destructor; virtual so that subclass destructors are invoked as well.
246 *
247 * @draft ICU 3.8
248 */
249 virtual ~ULocRuns();
250
251 /**
252 * Get the name of the locale assoicated with the given run
253 * of text. Use <code>RunArray::getLimit(run)</code> to get the corresponding
254 * limit index.
255 *
256 * @param run is the index into the font and limit indices arrays.
257 *
258 * @return the locale name associated with the given text run.
259 *
260 * @see RunArray::getLimit
261 *
262 * @draft ICU 3.8
263 */
264 const char *getLocaleName(le_int32 run) const;
265
266 /**
267 * Add a <code>Locale</code> and limit index pair to the data arrays and return
268 * the run index where the data was stored. This method calls
269 * <code>RunArray::add(limit)</code> which will create or grow the arrays as needed.
270 *
271 * If the <code>ULocRuns</code> object was created with a client-supplied
272 * locale and limit indices arrays, this method will return a run index of -1.
273 *
274 * Subclasses should not override this method. Rather they should provide a new <code>add</code>
275 * method which takes a locale name and a limit index along with whatever other data they implement.
276 * The new <code>add</code> method should first call this method to grow the font and limit indices
277 * arrays, and use the returned run index to store data their own arrays.
278 *
279 * @param locale is the name of the locale to add. This object must remain valid
280 * until the <code>ULocRuns</code> object is destroyed.
281 *
282 * @param limit is the limit index to add
283 *
284 * @return the run index where the locale and limit index were stored, or -1 if the data cannot be stored.
285 *
286 * @draft ICU 3.8
287 */
288 le_int32 add(const char *locale, le_int32 limit);
289
290 /**
291 * ICU "poor man's RTTI", returns a UClassID for this class.
292 *
293 * @draft ICU 3.8
294 */
295 static inline UClassID getStaticClassID();
296
297 /**
298 * ICU "poor man's RTTI", returns a UClassID for the actual class.
299 *
300 * @draft ICU 3.8
301 */
302 virtual inline UClassID getDynamicClassID() const;
303
304 protected:
305 virtual void init(le_int32 capacity);
306 virtual void grow(le_int32 capacity);
307
308 private:
309
310 inline ULocRuns();
311 inline ULocRuns(const ULocRuns &other);
operator =(const ULocRuns &)312 inline ULocRuns &operator=(const ULocRuns & /*other*/) { return *this; };
313 const char **fLocaleNames;
314 Locale **fLocalesCopy;
315 };
316
ULocRuns()317 inline ULocRuns::ULocRuns()
318 : LocaleRuns(0), fLocaleNames(NULL)
319 {
320 // nothing else to do...
321 }
322
ULocRuns(const ULocRuns &)323 inline ULocRuns::ULocRuns(const ULocRuns & /*other*/)
324 : LocaleRuns(0), fLocaleNames(NULL)
325 {
326 // nothing else to do...
327 }
328
getLocales(const char ** localeNames,le_int32 count)329 static const Locale **getLocales(const char **localeNames, le_int32 count)
330 {
331 Locale **locales = LE_NEW_ARRAY(Locale *, count);
332
333 for (int i = 0; i < count; i += 1) {
334 locales[i] = new Locale(Locale::createFromName(localeNames[i]));
335 }
336
337 return (const Locale **) locales;
338 }
339
ULocRuns(const char ** locales,const le_int32 * limits,le_int32 count)340 ULocRuns::ULocRuns(const char **locales, const le_int32 *limits, le_int32 count)
341 : LocaleRuns(getLocales(locales, count), limits, count), fLocaleNames(locales)
342 {
343 // nothing else to do...
344 }
345
ULocRuns(le_int32 initialCapacity)346 ULocRuns::ULocRuns(le_int32 initialCapacity)
347 : LocaleRuns(initialCapacity), fLocaleNames(NULL)
348 {
349 if(initialCapacity > 0) {
350 fLocaleNames = LE_NEW_ARRAY(const char *, initialCapacity);
351 }
352 }
353
~ULocRuns()354 ULocRuns::~ULocRuns()
355 {
356 le_int32 count = getCount();
357
358 for(int i = 0; i < count; i += 1) {
359 delete fLocales[i];
360 }
361
362 if (fClientArrays) {
363 LE_DELETE_ARRAY(fLocales);
364 fLocales = NULL;
365 } else {
366 LE_DELETE_ARRAY(fLocaleNames);
367 fLocaleNames = NULL;
368 }
369 }
370
init(le_int32 capacity)371 void ULocRuns::init(le_int32 capacity)
372 {
373 LocaleRuns::init(capacity);
374 fLocaleNames = LE_NEW_ARRAY(const char *, capacity);
375 }
376
grow(le_int32 capacity)377 void ULocRuns::grow(le_int32 capacity)
378 {
379 LocaleRuns::grow(capacity);
380 fLocaleNames = (const char **) LE_GROW_ARRAY(fLocaleNames, capacity);
381 }
382
add(const char * locale,le_int32 limit)383 le_int32 ULocRuns::add(const char *locale, le_int32 limit)
384 {
385 Locale *loc = new Locale(Locale::createFromName(locale));
386 le_int32 index = LocaleRuns::add(loc, limit);
387
388 if (index >= 0) {
389 char **localeNames = (char **) fLocaleNames;
390
391 localeNames[index] = (char *) locale;
392 }
393
394 return index;
395 }
396
getLocaleName(le_int32 run) const397 const char *ULocRuns::getLocaleName(le_int32 run) const
398 {
399 if (run < 0 || run >= getCount()) {
400 return NULL;
401 }
402
403 return fLocaleNames[run];
404 }
UOBJECT_DEFINE_RTTI_IMPLEMENTATION(ULocRuns)405 UOBJECT_DEFINE_RTTI_IMPLEMENTATION(ULocRuns)
406 U_NAMESPACE_END
407
408 U_CAPI pl_localeRuns * U_EXPORT2
409 pl_openLocaleRuns(const char **locales,
410 const le_int32 *limits,
411 le_int32 count)
412 {
413 return (pl_localeRuns *) new ULocRuns(locales, limits, count);
414 }
415
416 U_CAPI pl_localeRuns * U_EXPORT2
pl_openEmptyLocaleRuns(le_int32 initialCapacity)417 pl_openEmptyLocaleRuns(le_int32 initialCapacity)
418 {
419 return (pl_localeRuns *) new ULocRuns(initialCapacity);
420 }
421
422 U_CAPI void U_EXPORT2
pl_closeLocaleRuns(pl_localeRuns * localeRuns)423 pl_closeLocaleRuns(pl_localeRuns *localeRuns)
424 {
425 ULocRuns *lr = (ULocRuns *) localeRuns;
426
427 delete lr;
428 }
429
430 U_CAPI le_int32 U_EXPORT2
pl_getLocaleRunCount(const pl_localeRuns * localeRuns)431 pl_getLocaleRunCount(const pl_localeRuns *localeRuns)
432 {
433 const ULocRuns *lr = (const ULocRuns *) localeRuns;
434
435 if (lr == NULL) {
436 return -1;
437 }
438
439 return lr->getCount();
440 }
441
442 U_CAPI void U_EXPORT2
pl_resetLocaleRuns(pl_localeRuns * localeRuns)443 pl_resetLocaleRuns(pl_localeRuns *localeRuns)
444 {
445 ULocRuns *lr = (ULocRuns *) localeRuns;
446
447 if (lr != NULL) {
448 lr->reset();
449 }
450 }
451
452 U_CAPI le_int32 U_EXPORT2
pl_getLocaleRunLastLimit(const pl_localeRuns * localeRuns)453 pl_getLocaleRunLastLimit(const pl_localeRuns *localeRuns)
454 {
455 const ULocRuns *lr = (const ULocRuns *) localeRuns;
456
457 if (lr == NULL) {
458 return -1;
459 }
460
461 return lr->getLimit();
462 }
463
464 U_CAPI le_int32 U_EXPORT2
pl_getLocaleRunLimit(const pl_localeRuns * localeRuns,le_int32 run)465 pl_getLocaleRunLimit(const pl_localeRuns *localeRuns,
466 le_int32 run)
467 {
468 const ULocRuns *lr = (const ULocRuns *) localeRuns;
469
470 if (lr == NULL) {
471 return -1;
472 }
473
474 return lr->getLimit(run);
475 }
476
477 U_CAPI const char * U_EXPORT2
pl_getLocaleRunLocale(const pl_localeRuns * localeRuns,le_int32 run)478 pl_getLocaleRunLocale(const pl_localeRuns *localeRuns,
479 le_int32 run)
480 {
481 const ULocRuns *lr = (const ULocRuns *) localeRuns;
482
483 if (lr == NULL) {
484 return NULL;
485 }
486
487 return lr->getLocaleName(run);
488 }
489
490 U_CAPI le_int32 U_EXPORT2
pl_addLocaleRun(pl_localeRuns * localeRuns,const char * locale,le_int32 limit)491 pl_addLocaleRun(pl_localeRuns *localeRuns,
492 const char *locale,
493 le_int32 limit)
494 {
495 ULocRuns *lr = (ULocRuns *) localeRuns;
496
497 if (lr == NULL) {
498 return -1;
499 }
500
501 return lr->add(locale, limit);
502 }
503