1 package org.unicode.cldr.test;
2 
3 import java.util.List;
4 
5 import org.unicode.cldr.test.CheckCLDR.CheckStatus.Subtype;
6 import org.unicode.cldr.util.CLDRFile;
7 import org.unicode.cldr.util.XPathParts;
8 
9 import com.ibm.icu.lang.UCharacter;
10 import com.ibm.icu.text.BreakIterator;
11 import com.ibm.icu.util.ULocale;
12 
13 public class CheckCasing extends CheckCLDR {
14     public enum Case {
15         mixed, lowercase_words, titlecase_words, titlecase_firstword, verbatim;
forString(String input)16         public static Case forString(String input) {
17             return valueOf(input.replace('-', '_'));
18         }
19     };
20 
21     // remember to add this class to the list in CheckCLDR.getCheckAll
22     // to run just this test, on just locales starting with 'nl', use CheckCLDR with -fnl.* -t.*Currencies.*
23 
24     XPathParts parts = new XPathParts(); // used to parse out a path
25     ULocale uLocale = null;
26     BreakIterator breaker = null;
27 
28     @Override
setCldrFileToCheck(CLDRFile cldrFileToCheck, Options options, List<CheckStatus> possibleErrors)29     public CheckCLDR setCldrFileToCheck(CLDRFile cldrFileToCheck, Options options,
30         List<CheckStatus> possibleErrors) {
31         if (cldrFileToCheck == null) return this;
32         super.setCldrFileToCheck(cldrFileToCheck, options, possibleErrors);
33         uLocale = new ULocale(cldrFileToCheck.getLocaleID());
34         breaker = BreakIterator.getWordInstance(uLocale);
35         return this;
36     }
37 
38     // If you don't need any file initialization or postprocessing, you only need this one routine
handleCheck(String path, String fullPath, String value, Options options, List<CheckStatus> result)39     public CheckCLDR handleCheck(String path, String fullPath, String value, Options options,
40         List<CheckStatus> result) {
41         // it helps performance to have a quick reject of most paths
42         if (fullPath == null) return this; // skip paths that we don't have
43         if (fullPath.indexOf("casing") < 0) return this;
44 
45         // pick up the casing attributes from the full path
46         parts.set(fullPath);
47 
48         Case caseTest = Case.mixed;
49         for (int i = 0; i < parts.size(); ++i) {
50             String casingValue = parts.getAttributeValue(i, "casing");
51             if (casingValue == null) {
52                 continue;
53             }
54             caseTest = Case.forString(casingValue);
55             if (caseTest == Case.verbatim) {
56                 return this; // we're done
57             }
58         }
59 
60         String newValue = value;
61         switch (caseTest) {
62         case lowercase_words:
63             newValue = UCharacter.toLowerCase(uLocale, value);
64             break;
65         case titlecase_words:
66             newValue = UCharacter.toTitleCase(uLocale, value, null);
67             break;
68         case titlecase_firstword:
69             newValue = TitleCaseFirst(uLocale, value);
70             break;
71         default:
72             break;
73 
74         }
75         if (!newValue.equals(value)) {
76             // the following is how you signal an error or warning (or add a demo....)
77             result.add(new CheckStatus().setCause(this)
78                 .setMainType(CheckStatus.errorType)
79                 .setSubtype(Subtype.incorrectCasing)
80                 // typically warningType or errorType
81                 .setMessage("Casing incorrect: either should have casing=\"verbatim\" or be <{0}>",
82                     new Object[] { newValue })); // the message; can be MessageFormat with arguments
83         }
84         return this;
85     }
86 
87     // -f(bg|cs|da|el|et|is|it|lt|ro|ru|sl|uk) -t(.*casing.*)
88 
TitleCaseFirst(ULocale locale, String value)89     private String TitleCaseFirst(ULocale locale, String value) {
90         if (value.length() == 0) {
91             return value;
92         }
93         breaker.setText(value);
94         breaker.first();
95         int endOfFirstWord = breaker.next();
96         return UCharacter.toTitleCase(uLocale, value.substring(0, endOfFirstWord), breaker)
97             + value.substring(endOfFirstWord);
98     }
99 
100 }