From 42b406bf178cdbd8098bf49fb88b9d8e06146862 Mon Sep 17 00:00:00 2001 From: Robin Leroy Date: Mon, 27 Jan 2025 01:56:54 +0100 Subject: [PATCH] ICU-22984 Generate old Java monkeys --- .../ibm/icu/dev/test/rbbi/RBBITestMonkey.java | 1290 +++-------------- .../com/ibm/icu/dev/test/rbbi/RegexRule.java | 111 ++ .../com/ibm/icu/dev/test/rbbi/RemapRule.java | 166 +++ .../icu/dev/test/rbbi/SegmentationRule.java | 94 ++ 4 files changed, 597 insertions(+), 1064 deletions(-) create mode 100644 icu4j/main/core/src/test/java/com/ibm/icu/dev/test/rbbi/RegexRule.java create mode 100644 icu4j/main/core/src/test/java/com/ibm/icu/dev/test/rbbi/RemapRule.java create mode 100644 icu4j/main/core/src/test/java/com/ibm/icu/dev/test/rbbi/SegmentationRule.java diff --git a/icu4j/main/core/src/test/java/com/ibm/icu/dev/test/rbbi/RBBITestMonkey.java b/icu4j/main/core/src/test/java/com/ibm/icu/dev/test/rbbi/RBBITestMonkey.java index 8d3ead20fb61..e80f9fce03b5 100644 --- a/icu4j/main/core/src/test/java/com/ibm/icu/dev/test/rbbi/RBBITestMonkey.java +++ b/icu4j/main/core/src/test/java/com/ibm/icu/dev/test/rbbi/RBBITestMonkey.java @@ -19,10 +19,12 @@ import org.junit.runners.JUnit4; // Monkey testing of RuleBasedBreakIterator. -// The old, original monkey test. TODO: remove +// The old monkey test, now using regexes generated by the Unicode tools. // The new monkey test is class RBBIMonkeyTest. import com.ibm.icu.dev.test.CoreTestFmwk; +import com.ibm.icu.dev.test.rbbi.SegmentationRule.BreakContext; +import com.ibm.icu.dev.test.rbbi.SegmentationRule.Resolution; import com.ibm.icu.lang.UCharacter; import com.ibm.icu.lang.UProperty; import com.ibm.icu.text.BreakIterator; @@ -676,166 +678,15 @@ int next(int prevPos) { static class RBBILineMonkey extends RBBIMonkeyKind { - // UnicodeSets for each of the Line Breaking character classes. - // Order matches that of Unicode UAX 14, Table 1, which makes it a little easier - // to verify that they are all accounted for. - - // XUnicodeSet is like UnicodeSet, except that the method contains(int codePoint) does not - // throw exceptions on out-of-range codePoints. This matches ICU4C behavior. - // The LineMonkey test (ported from ICU4C) relies on this behavior, it uses a value of -1 - // to represent a non-codepoint that is not included in any of the property sets. - // This happens for rule 30a. - class XUnicodeSet extends UnicodeSet { - XUnicodeSet(String pattern) { super(pattern); } - XUnicodeSet() { super(); } - @Override - public boolean contains(int codePoint) { - return codePoint < UnicodeSet.MIN_VALUE || codePoint > UnicodeSet.MAX_VALUE ? - false : super.contains(codePoint); - } - } - - // Declare these variables as XUnicodeSet, not merely as UnicodeSet, - // so that when we copy a new declaration from C++ (where only UnicodeSet exists), - // the missing 'X' prefix is visible; - // and when the prefix is there and we copy a new initializer we get a compiler error. - // (Otherwise we rely on the caller catching the IAE from using codePoint=-1 - // and failing with a message that tells us what to do.) - XUnicodeSet fBK; - XUnicodeSet fCR; - XUnicodeSet fLF; - XUnicodeSet fCM; - XUnicodeSet fNL; - XUnicodeSet fSG; - XUnicodeSet fWJ; - XUnicodeSet fZW; - XUnicodeSet fGL; - XUnicodeSet fSP; - XUnicodeSet fB2; - XUnicodeSet fBA; - XUnicodeSet fBB; - XUnicodeSet fHH; - XUnicodeSet fHY; - XUnicodeSet fCB; - XUnicodeSet fCL; - XUnicodeSet fCP; - XUnicodeSet fEX; - XUnicodeSet fIN; - XUnicodeSet fNS; - XUnicodeSet fOP; - XUnicodeSet fQU; - XUnicodeSet fIS; - XUnicodeSet fNU; - XUnicodeSet fPO; - XUnicodeSet fPR; - XUnicodeSet fSY; - XUnicodeSet fAI; - XUnicodeSet fAL; - XUnicodeSet fCJ; - XUnicodeSet fH2; - XUnicodeSet fH3; - XUnicodeSet fHL; - XUnicodeSet fID; - XUnicodeSet fJL; - XUnicodeSet fJV; - XUnicodeSet fJT; - XUnicodeSet fRI; - XUnicodeSet fXX; - XUnicodeSet fEB; - XUnicodeSet fEM; - XUnicodeSet fZWJ; - XUnicodeSet fOP30; - XUnicodeSet fCP30; - XUnicodeSet fExtPictUnassigned; - XUnicodeSet fAK; - XUnicodeSet fAP; - XUnicodeSet fAS; - XUnicodeSet fVF; - XUnicodeSet fVI; - XUnicodeSet fPi; - XUnicodeSet fPf; - XUnicodeSet feaFWH; + List rules; + SegmentationRule.BreakContext[] resolved; StringBuffer fText; - int fOrigPositions; RBBILineMonkey() { fCharProperty = UProperty.LINE_BREAK; - fBK = new XUnicodeSet("[\\p{Line_Break=BK}]"); - fCR = new XUnicodeSet("[\\p{Line_break=CR}]"); - fLF = new XUnicodeSet("[\\p{Line_break=LF}]"); - fCM = new XUnicodeSet("[\\p{Line_break=CM}]"); - fNL = new XUnicodeSet("[\\p{Line_break=NL}]"); - fSG = new XUnicodeSet("[\\ud800-\\udfff]"); - fWJ = new XUnicodeSet("[\\p{Line_break=WJ}]"); - fZW = new XUnicodeSet("[\\p{Line_break=ZW}]"); - fGL = new XUnicodeSet("[\\p{Line_break=GL}]"); - fSP = new XUnicodeSet("[\\p{Line_break=SP}]"); - fB2 = new XUnicodeSet("[\\p{Line_break=B2}]"); - fBA = new XUnicodeSet("[\\p{Line_break=BA}]"); - fBB = new XUnicodeSet("[\\p{Line_break=BB}]"); - fHH = new XUnicodeSet(); - fHY = new XUnicodeSet("[\\p{Line_break=HY}]"); - fCB = new XUnicodeSet("[\\p{Line_break=CB}]"); - fCL = new XUnicodeSet("[\\p{Line_break=CL}]"); - fCP = new XUnicodeSet("[\\p{Line_break=CP}]"); - fEX = new XUnicodeSet("[\\p{Line_break=EX}]"); - fIN = new XUnicodeSet("[\\p{Line_break=IN}]"); - fNS = new XUnicodeSet("[\\p{Line_break=NS}]"); - fOP = new XUnicodeSet("[\\p{Line_break=OP}]"); - fQU = new XUnicodeSet("[\\p{Line_break=QU}]"); - fIS = new XUnicodeSet("[\\p{Line_break=IS}]"); - fNU = new XUnicodeSet("[\\p{Line_break=NU}]"); - fPO = new XUnicodeSet("[\\p{Line_break=PO}]"); - fPR = new XUnicodeSet("[\\p{Line_break=PR}]"); - fSY = new XUnicodeSet("[\\p{Line_break=SY}]"); - fAI = new XUnicodeSet("[\\p{Line_break=AI}]"); - fAL = new XUnicodeSet("[\\p{Line_break=AL}]"); - fCJ = new XUnicodeSet("[\\p{Line_break=CJ}]"); - fH2 = new XUnicodeSet("[\\p{Line_break=H2}]"); - fH3 = new XUnicodeSet("[\\p{Line_break=H3}]"); - fHL = new XUnicodeSet("[\\p{Line_break=HL}]"); - fID = new XUnicodeSet("[\\p{Line_break=ID}]"); - fJL = new XUnicodeSet("[\\p{Line_break=JL}]"); - fJV = new XUnicodeSet("[\\p{Line_break=JV}]"); - fJT = new XUnicodeSet("[\\p{Line_break=JT}]"); - fRI = new XUnicodeSet("[\\p{Line_break=RI}]"); - fXX = new XUnicodeSet("[\\p{Line_break=XX}]"); - fEB = new XUnicodeSet("[\\p{Line_break=EB}]"); - fEM = new XUnicodeSet("[\\p{Line_break=EM}]"); - fZWJ = new XUnicodeSet("[\\p{Line_break=ZWJ}]"); - fOP30 = new XUnicodeSet("[\\p{Line_break=OP}-[\\p{ea=F}\\p{ea=W}\\p{ea=H}]]"); - fCP30 = new XUnicodeSet("[\\p{Line_break=CP}-[\\p{ea=F}\\p{ea=W}\\p{ea=H}]]"); - fExtPictUnassigned = new XUnicodeSet("[\\p{Extended_Pictographic}&\\p{Cn}]"); - fAK = new XUnicodeSet("[\\p{Line_Break=AK}]"); - fAP = new XUnicodeSet("[\\p{Line_Break=AP}]"); - fAS = new XUnicodeSet("[\\p{Line_Break=AS}]"); - fVF = new XUnicodeSet("[\\p{Line_Break=VF}]"); - fVI = new XUnicodeSet("[\\p{Line_Break=VI}]"); - - fPi = new XUnicodeSet("[\\p{Pi}]"); - fPf = new XUnicodeSet("[\\p{Pf}]"); - - feaFWH = new XUnicodeSet("[\\p{ea=F}\\p{ea=W}\\p{ea=H}]"); - - // Remove dictionary characters. - // The monkey test reference implementation of line break does not replicate the dictionary behavior, - // so dictionary characters are omitted from the monkey test data. - @SuppressWarnings("unused") - UnicodeSet dictionarySet = new UnicodeSet( - "[[:LineBreak = Complex_Context:] & [[:Script = Thai:][:Script = Lao:][:Script = Khmer:] [:script = Myanmar:]]]"); - - fAL.addAll(fXX); // Default behavior for XX is identical to AL - fAL.addAll(fAI); // Default behavior for AI is identical to AL - fAL.addAll(fSG); // Default behavior for SG (unpaired surrogates) is AL - - fNS.addAll(fCJ); // Default behavior for CJ is identical to NS. - fCM.addAll(fZWJ); // ZWJ behaves as a CM. - - fHH.add('\u2010'); // Hyphen, '‐' - class NamedSet { String name; UnicodeSet set; @@ -847,40 +698,181 @@ class NamedSet { this(name, new UnicodeSet(pattern)); } }; - - final List interestingSets = new ArrayList<>(); - interestingSets.add(new NamedSet("eastAsian", "[\\p{ea=F}\\p{ea=W}\\p{ea=H}]")); - interestingSets.add(new NamedSet("Pi", "\\p{Pi}")); - interestingSets.add(new NamedSet("Pf", "\\p{Pf}")); - interestingSets.add(new NamedSet("DOTTEDC.", "[◌]")); - interestingSets.add(new NamedSet("HYPHEN", "[\\u2010]")); - interestingSets.add(new NamedSet("ExtPictCn", "[\\p{Extended_Pictographic}&\\p{Cn}]")); - final List partition = new ArrayList<>(); - for (int lb = 0; lb < UCharacter.LineBreak.COUNT; ++lb) { - final String lbValueShortName = - UCharacter.getPropertyValueName(UProperty.LINE_BREAK, lb, UProperty.NameChoice.SHORT); - if (lbValueShortName.equals("SA")) { + List partition = new ArrayList<>(); + rules = new ArrayList<>(); + + rules.add(new RegexRule("sot ÷ contra LB2", "^", Resolution.BREAK, "")); + // This one could be part of the rules. + // Note that /$/ matches ( BK | CR | LF | NL ) eot, so we use (?!.) instead. + // The generated rules use the same (?!.). + rules.add(new RegexRule("LB3 ÷ eot", "", Resolution.BREAK, "(?!.)")); + + // --- NOLI ME TANGERE --- + // Generated by GenerateBreakTest.java in the Unicode tools. + partition.add(new NamedSet("AI_EastAsian", new UnicodeSet("[\\p{Line_Break=Ambiguous}&[\\p{ea=F}\\p{ea=W}\\p{ea=H}]]"))); + partition.add(new NamedSet("AImEastAsian", new UnicodeSet("[\\p{Line_Break=Ambiguous}-[\\p{ea=F}\\p{ea=W}\\p{ea=H}]]"))); + partition.add(new NamedSet("AK", new UnicodeSet("[\\p{Line_Break=Aksara}]"))); + partition.add(new NamedSet("ALorig_EastAsian", new UnicodeSet("[\\p{Line_Break=Alphabetic}&[\\p{ea=F}\\p{ea=W}\\p{ea=H}]]"))); + partition.add(new NamedSet("ALorig_DottedCircle", new UnicodeSet("[\\p{Line_Break=Alphabetic}&[◌]]"))); + partition.add(new NamedSet("ALorigmEastAsianmDottedCircle", new UnicodeSet("[\\p{Line_Break=Alphabetic}-[\\p{ea=F}\\p{ea=W}\\p{ea=H}]-[◌]]"))); + partition.add(new NamedSet("AP", new UnicodeSet("[\\p{Line_Break=Aksara_Prebase}]"))); + partition.add(new NamedSet("AS", new UnicodeSet("[\\p{Line_Break=Aksara_Start}]"))); + partition.add(new NamedSet("B2", new UnicodeSet("[\\p{Line_Break=Break_Both}]"))); + partition.add(new NamedSet("BA_EastAsian", new UnicodeSet("[\\p{Line_Break=Break_After}&[\\p{ea=F}\\p{ea=W}\\p{ea=H}]]"))); + partition.add(new NamedSet("BA_Hyphen", new UnicodeSet("[\\p{Line_Break=Break_After}&[\\u2010]]"))); + partition.add(new NamedSet("BAmEastAsianmHyphen", new UnicodeSet("[\\p{Line_Break=Break_After}-[\\p{ea=F}\\p{ea=W}\\p{ea=H}]-[\\u2010]]"))); + partition.add(new NamedSet("BB", new UnicodeSet("[\\p{Line_Break=Break_Before}]"))); + partition.add(new NamedSet("BK", new UnicodeSet("[\\p{Line_Break=Mandatory_Break}]"))); + partition.add(new NamedSet("CB", new UnicodeSet("[\\p{Line_Break=Contingent_Break}]"))); + partition.add(new NamedSet("CL_EastAsian", new UnicodeSet("[\\p{Line_Break=Close_Punctuation}&[\\p{ea=F}\\p{ea=W}\\p{ea=H}]]"))); + partition.add(new NamedSet("CLmEastAsian", new UnicodeSet("[\\p{Line_Break=Close_Punctuation}-[\\p{ea=F}\\p{ea=W}\\p{ea=H}]]"))); + partition.add(new NamedSet("CP", new UnicodeSet("[\\p{Line_Break=CP}]"))); + partition.add(new NamedSet("CMorig_EastAsian", new UnicodeSet("[\\p{Line_Break=Combining_Mark}&[\\p{ea=F}\\p{ea=W}\\p{ea=H}]]"))); + partition.add(new NamedSet("CMorigmEastAsian", new UnicodeSet("[\\p{Line_Break=Combining_Mark}-[\\p{ea=F}\\p{ea=W}\\p{ea=H}]]"))); + partition.add(new NamedSet("CR", new UnicodeSet("[\\p{Line_Break=Carriage_Return}]"))); + partition.add(new NamedSet("EX_EastAsian", new UnicodeSet("[\\p{Line_Break=Exclamation}&[\\p{ea=F}\\p{ea=W}\\p{ea=H}]]"))); + partition.add(new NamedSet("EXmEastAsian", new UnicodeSet("[\\p{Line_Break=Exclamation}-[\\p{ea=F}\\p{ea=W}\\p{ea=H}]]"))); + partition.add(new NamedSet("GL_EastAsian", new UnicodeSet("[\\p{Line_Break=Glue}&[\\p{ea=F}\\p{ea=W}\\p{ea=H}]]"))); + partition.add(new NamedSet("GLmEastAsian", new UnicodeSet("[\\p{Line_Break=Glue}-[\\p{ea=F}\\p{ea=W}\\p{ea=H}]]"))); + partition.add(new NamedSet("H2", new UnicodeSet("[\\p{Line_Break=H2}]"))); + partition.add(new NamedSet("H3", new UnicodeSet("[\\p{Line_Break=H3}]"))); + partition.add(new NamedSet("HL", new UnicodeSet("[\\p{Line_Break=HL}]"))); + partition.add(new NamedSet("HY", new UnicodeSet("[\\p{Line_Break=Hyphen}]"))); + partition.add(new NamedSet("ID_EastAsian", new UnicodeSet("[\\p{Line_Break=Ideographic}&[\\p{ea=F}\\p{ea=W}\\p{ea=H}]]"))); + partition.add(new NamedSet("ID_ExtPictUnassigned", new UnicodeSet("[\\p{Line_Break=Ideographic}&[\\p{Extended_Pictographic}&\\p{gc=Cn}]]"))); + partition.add(new NamedSet("IDmEastAsianmExtPictUnassigned", new UnicodeSet("[\\p{Line_Break=Ideographic}-[\\p{ea=F}\\p{ea=W}\\p{ea=H}]-[\\p{Extended_Pictographic}&\\p{gc=Cn}]]"))); + partition.add(new NamedSet("IN_EastAsian", new UnicodeSet("[\\p{Line_Break=Inseparable}&[\\p{ea=F}\\p{ea=W}\\p{ea=H}]]"))); + partition.add(new NamedSet("INmEastAsian", new UnicodeSet("[\\p{Line_Break=Inseparable}-[\\p{ea=F}\\p{ea=W}\\p{ea=H}]]"))); + partition.add(new NamedSet("IS", new UnicodeSet("[\\p{Line_Break=Infix_Numeric}]"))); + partition.add(new NamedSet("JL", new UnicodeSet("[\\p{Line_Break=JL}]"))); + partition.add(new NamedSet("JT", new UnicodeSet("[\\p{Line_Break=JT}]"))); + partition.add(new NamedSet("JV", new UnicodeSet("[\\p{Line_Break=JV}]"))); + partition.add(new NamedSet("LF", new UnicodeSet("[\\p{Line_Break=Line_Feed}]"))); + partition.add(new NamedSet("NL", new UnicodeSet("[\\p{Line_Break=Next_Line}]"))); + partition.add(new NamedSet("NSorig_EastAsian", new UnicodeSet("[\\p{Line_Break=Nonstarter}&[\\p{ea=F}\\p{ea=W}\\p{ea=H}]]"))); + partition.add(new NamedSet("NSorigmEastAsian", new UnicodeSet("[\\p{Line_Break=Nonstarter}-[\\p{ea=F}\\p{ea=W}\\p{ea=H}]]"))); + partition.add(new NamedSet("NU", new UnicodeSet("[\\p{Line_Break=Numeric}]"))); + partition.add(new NamedSet("OP_EastAsian", new UnicodeSet("[\\p{Line_Break=Open_Punctuation}&[\\p{ea=F}\\p{ea=W}\\p{ea=H}]]"))); + partition.add(new NamedSet("OPmEastAsian", new UnicodeSet("[\\p{Line_Break=Open_Punctuation}-[\\p{ea=F}\\p{ea=W}\\p{ea=H}]]"))); + partition.add(new NamedSet("PO_EastAsian", new UnicodeSet("[\\p{Line_Break=Postfix_Numeric}&[\\p{ea=F}\\p{ea=W}\\p{ea=H}]]"))); + partition.add(new NamedSet("POmEastAsian", new UnicodeSet("[\\p{Line_Break=Postfix_Numeric}-[\\p{ea=F}\\p{ea=W}\\p{ea=H}]]"))); + partition.add(new NamedSet("PR_EastAsian", new UnicodeSet("[\\p{Line_Break=Prefix_Numeric}&[\\p{ea=F}\\p{ea=W}\\p{ea=H}]]"))); + partition.add(new NamedSet("PRmEastAsian", new UnicodeSet("[\\p{Line_Break=Prefix_Numeric}-[\\p{ea=F}\\p{ea=W}\\p{ea=H}]]"))); + partition.add(new NamedSet("QU_Pi", new UnicodeSet("[\\p{Line_Break=Quotation}&\\p{gc=Pi}]"))); + partition.add(new NamedSet("QU_Pf", new UnicodeSet("[\\p{Line_Break=Quotation}&\\p{gc=Pf}]"))); + partition.add(new NamedSet("QUmPimPf", new UnicodeSet("[\\p{Line_Break=Quotation}-\\p{gc=Pi}-\\p{gc=Pf}]"))); + partition.add(new NamedSet("SA_Mn", new UnicodeSet("[[\\p{Line_Break=Complex_Context}&\\p{gc=Mn}]]"))); + partition.add(new NamedSet("SA_Mc", new UnicodeSet("[[\\p{Line_Break=Complex_Context}&\\p{gc=Mc}]]"))); + partition.add(new NamedSet("SAmMnmMc", new UnicodeSet("[[\\p{Line_Break=Complex_Context}-\\p{gc=Mn}-\\p{gc=Mc}]]"))); + partition.add(new NamedSet("SG", new UnicodeSet("[\\p{Line_Break=Surrogate}]"))); + partition.add(new NamedSet("SP", new UnicodeSet("[\\p{Line_Break=Space}]"))); + partition.add(new NamedSet("SY", new UnicodeSet("[\\p{Line_Break=Break_Symbols}]"))); + partition.add(new NamedSet("VF", new UnicodeSet("[\\p{Line_Break=Virama_Final}]"))); + partition.add(new NamedSet("VI", new UnicodeSet("[\\p{Line_Break=Virama}]"))); + partition.add(new NamedSet("WJ", new UnicodeSet("[\\p{Line_Break=Word_Joiner}]"))); + partition.add(new NamedSet("XX_ExtPictUnassigned", new UnicodeSet("[\\p{Line_Break=Unknown}&[\\p{Extended_Pictographic}&\\p{gc=Cn}]]"))); + partition.add(new NamedSet("XXmExtPictUnassigned", new UnicodeSet("[\\p{Line_Break=Unknown}-[\\p{Extended_Pictographic}&\\p{gc=Cn}]]"))); + partition.add(new NamedSet("ZW", new UnicodeSet("[\\p{Line_Break=ZWSpace}]"))); + partition.add(new NamedSet("CJ", new UnicodeSet("[\\p{Line_Break=Conditional_Japanese_Starter}]"))); + partition.add(new NamedSet("RI", new UnicodeSet("[\\p{Line_Break=Regional_Indicator}]"))); + partition.add(new NamedSet("EB_EastAsian", new UnicodeSet("[\\p{Line_Break=E_Base}&[\\p{ea=F}\\p{ea=W}\\p{ea=H}]]"))); + partition.add(new NamedSet("EBmEastAsian", new UnicodeSet("[\\p{Line_Break=E_Base}-[\\p{ea=F}\\p{ea=W}\\p{ea=H}]]"))); + partition.add(new NamedSet("EM", new UnicodeSet("[\\p{Line_Break=E_Modifier}]"))); + partition.add(new NamedSet("ZWJ", new UnicodeSet("[\\p{Line_Break=ZWJ}]"))); + + rules.add(new RegexRule("$BK ÷", "\\p{Line_Break=Mandatory_Break}", Resolution.BREAK, "")); + rules.add(new RegexRule("$CR × $LF", "\\p{Line_Break=Carriage_Return}", Resolution.NO_BREAK, "\\p{Line_Break=Line_Feed}")); + rules.add(new RegexRule("$CR ÷", "\\p{Line_Break=Carriage_Return}", Resolution.BREAK, "")); + rules.add(new RegexRule("$LF ÷", "\\p{Line_Break=Line_Feed}", Resolution.BREAK, "")); + rules.add(new RegexRule("$NL ÷", "\\p{Line_Break=Next_Line}", Resolution.BREAK, "")); + rules.add(new RegexRule("× ( $BK | $CR | $LF | $NL )", "", Resolution.NO_BREAK, "( \\p{Line_Break=Mandatory_Break} | \\p{Line_Break=Carriage_Return} | \\p{Line_Break=Line_Feed} | \\p{Line_Break=Next_Line} )")); + rules.add(new RegexRule("× $SP", "", Resolution.NO_BREAK, "\\p{Line_Break=Space}")); + rules.add(new RegexRule("× $ZW", "", Resolution.NO_BREAK, "\\p{Line_Break=ZWSpace}")); + rules.add(new RegexRule("$ZW $SP* ÷", "\\p{Line_Break=ZWSpace} \\p{Line_Break=Space}*", Resolution.BREAK, "")); + rules.add(new RegexRule("$ZWJ ×", "\\p{Line_Break=ZWJ}", Resolution.NO_BREAK, "")); + rules.add(new RemapRule("(?[^$BK $CR $LF $NL $SP $ZW]) ( $CM | $ZWJ )* → ${X}", "(?[^\\p{Line_Break=Mandatory_Break} \\p{Line_Break=Carriage_Return} \\p{Line_Break=Line_Feed} \\p{Line_Break=Next_Line} \\p{Line_Break=Space} \\p{Line_Break=ZWSpace}]) ( [\\p{Line_Break=Combining_Mark} [\\p{Line_Break=Complex_Context}&\\p{gc=Mn}] [\\p{Line_Break=Complex_Context}&\\p{gc=Mc}]] | \\p{Line_Break=ZWJ} )*", "${X}")); + rules.add(new RemapRule("( $CM | $ZWJ ) → A", "( [\\p{Line_Break=Combining_Mark} [\\p{Line_Break=Complex_Context}&\\p{gc=Mn}] [\\p{Line_Break=Complex_Context}&\\p{gc=Mc}]] | \\p{Line_Break=ZWJ} )", "A")); + rules.add(new RegexRule("× $WJ", "", Resolution.NO_BREAK, "\\p{Line_Break=Word_Joiner}")); + rules.add(new RegexRule("$WJ ×", "\\p{Line_Break=Word_Joiner}", Resolution.NO_BREAK, "")); + rules.add(new RegexRule("$GL ×", "\\p{Line_Break=Glue}", Resolution.NO_BREAK, "")); + rules.add(new RegexRule("[^ $SP $BA $HY] × $GL", "[^ \\p{Line_Break=Space} \\p{Line_Break=Break_After} \\p{Line_Break=Hyphen}]", Resolution.NO_BREAK, "\\p{Line_Break=Glue}")); + rules.add(new RegexRule("× $EX", "", Resolution.NO_BREAK, "\\p{Line_Break=Exclamation}")); + rules.add(new RegexRule("× $CL", "", Resolution.NO_BREAK, "\\p{Line_Break=Close_Punctuation}")); + rules.add(new RegexRule("× $CP", "", Resolution.NO_BREAK, "\\p{Line_Break=CP}")); + rules.add(new RegexRule("× $SY", "", Resolution.NO_BREAK, "\\p{Line_Break=Break_Symbols}")); + rules.add(new RegexRule("$OP $SP* ×", "\\p{Line_Break=Open_Punctuation} \\p{Line_Break=Space}*", Resolution.NO_BREAK, "")); + rules.add(new RegexRule("( $BK | $CR | $LF | $NL | $OP | $QU | $GL | $SP | $ZW | $sot ) $QU_Pi $SP* ×", "( \\p{Line_Break=Mandatory_Break} | \\p{Line_Break=Carriage_Return} | \\p{Line_Break=Line_Feed} | \\p{Line_Break=Next_Line} | \\p{Line_Break=Open_Punctuation} | \\p{Line_Break=Quotation} | \\p{Line_Break=Glue} | \\p{Line_Break=Space} | \\p{Line_Break=ZWSpace} | ^ ) [\\p{Line_Break=Quotation} & \\p{gc=Pi}] \\p{Line_Break=Space}*", Resolution.NO_BREAK, "")); + rules.add(new RegexRule("× $QU_Pf ( $SP | $GL | $WJ | $CL | $QU | $CP | $EX | $IS | $SY | $BK | $CR | $LF | $NL | $ZW | $eot )", "", Resolution.NO_BREAK, "[\\p{Line_Break=Quotation} & \\p{gc=Pf}] ( \\p{Line_Break=Space} | \\p{Line_Break=Glue} | \\p{Line_Break=Word_Joiner} | \\p{Line_Break=Close_Punctuation} | \\p{Line_Break=Quotation} | \\p{Line_Break=CP} | \\p{Line_Break=Exclamation} | \\p{Line_Break=Infix_Numeric} | \\p{Line_Break=Break_Symbols} | \\p{Line_Break=Mandatory_Break} | \\p{Line_Break=Carriage_Return} | \\p{Line_Break=Line_Feed} | \\p{Line_Break=Next_Line} | \\p{Line_Break=ZWSpace} | (?!.) )")); + rules.add(new RegexRule("$SP ÷ $IS $NU", "\\p{Line_Break=Space}", Resolution.BREAK, "\\p{Line_Break=Infix_Numeric} \\p{Line_Break=Numeric}")); + rules.add(new RegexRule("× $IS", "", Resolution.NO_BREAK, "\\p{Line_Break=Infix_Numeric}")); + rules.add(new RegexRule("($CL | $CP) $SP* × $NS", "(\\p{Line_Break=Close_Punctuation} | \\p{Line_Break=CP}) \\p{Line_Break=Space}*", Resolution.NO_BREAK, "[\\p{Line_Break=Nonstarter} \\p{Line_Break=Conditional_Japanese_Starter}]")); + rules.add(new RegexRule("$B2 $SP* × $B2", "\\p{Line_Break=Break_Both} \\p{Line_Break=Space}*", Resolution.NO_BREAK, "\\p{Line_Break=Break_Both}")); + rules.add(new RegexRule("$SP ÷", "\\p{Line_Break=Space}", Resolution.BREAK, "")); + rules.add(new RegexRule("× $QUmPi", "", Resolution.NO_BREAK, "[\\p{Line_Break=Quotation} - \\p{gc=Pi}]")); + rules.add(new RegexRule("$QUmPf ×", "[\\p{Line_Break=Quotation} - \\p{gc=Pf}]", Resolution.NO_BREAK, "")); + rules.add(new RegexRule("[^$EastAsian] × $QU", "[^[\\p{ea=F}\\p{ea=W}\\p{ea=H}]]", Resolution.NO_BREAK, "\\p{Line_Break=Quotation}")); + rules.add(new RegexRule("× $QU ( [^$EastAsian] | $eot )", "", Resolution.NO_BREAK, "\\p{Line_Break=Quotation} ( [^[\\p{ea=F}\\p{ea=W}\\p{ea=H}]] | (?!.) )")); + rules.add(new RegexRule("$QU × [^$EastAsian]", "\\p{Line_Break=Quotation}", Resolution.NO_BREAK, "[^[\\p{ea=F}\\p{ea=W}\\p{ea=H}]]")); + rules.add(new RegexRule("( [^$EastAsian] | $sot ) $QU ×", "( [^[\\p{ea=F}\\p{ea=W}\\p{ea=H}]] | ^ ) \\p{Line_Break=Quotation}", Resolution.NO_BREAK, "")); + rules.add(new RegexRule("÷ $CB", "", Resolution.BREAK, "\\p{Line_Break=Contingent_Break}")); + rules.add(new RegexRule("$CB ÷", "\\p{Line_Break=Contingent_Break}", Resolution.BREAK, "")); + rules.add(new RegexRule("( $BK | $CR | $LF | $NL | $SP | $ZW | $CB | $GL | $sot ) ( $HY | $Hyphen ) × $AL", "( \\p{Line_Break=Mandatory_Break} | \\p{Line_Break=Carriage_Return} | \\p{Line_Break=Line_Feed} | \\p{Line_Break=Next_Line} | \\p{Line_Break=Space} | \\p{Line_Break=ZWSpace} | \\p{Line_Break=Contingent_Break} | \\p{Line_Break=Glue} | ^ ) ( \\p{Line_Break=Hyphen} | [\\u2010] )", Resolution.NO_BREAK, "[\\p{Line_Break=Ambiguous} \\p{Line_Break=Alphabetic} \\p{Line_Break=Surrogate} \\p{Line_Break=Unknown} [\\p{Line_Break=Complex_Context}-\\p{gc=Mn}-\\p{gc=Mc}]]")); + rules.add(new RegexRule("× $BA", "", Resolution.NO_BREAK, "\\p{Line_Break=Break_After}")); + rules.add(new RegexRule("× $HY", "", Resolution.NO_BREAK, "\\p{Line_Break=Hyphen}")); + rules.add(new RegexRule("× $NS", "", Resolution.NO_BREAK, "[\\p{Line_Break=Nonstarter} \\p{Line_Break=Conditional_Japanese_Starter}]")); + rules.add(new RegexRule("$BB ×", "\\p{Line_Break=Break_Before}", Resolution.NO_BREAK, "")); + rules.add(new RegexRule("$HL ($HY | $NonEastAsianBA) × [^$HL]", "\\p{Line_Break=HL} (\\p{Line_Break=Hyphen} | [\\p{Line_Break=Break_After} & [^[\\p{ea=F}\\p{ea=W}\\p{ea=H}]]])", Resolution.NO_BREAK, "[^\\p{Line_Break=HL}]")); + rules.add(new RegexRule("$SY × $HL", "\\p{Line_Break=Break_Symbols}", Resolution.NO_BREAK, "\\p{Line_Break=HL}")); + rules.add(new RegexRule("× $IN", "", Resolution.NO_BREAK, "\\p{Line_Break=Inseparable}")); + rules.add(new RegexRule("($AL | $HL) × $NU", "([\\p{Line_Break=Ambiguous} \\p{Line_Break=Alphabetic} \\p{Line_Break=Surrogate} \\p{Line_Break=Unknown} [\\p{Line_Break=Complex_Context}-\\p{gc=Mn}-\\p{gc=Mc}]] | \\p{Line_Break=HL})", Resolution.NO_BREAK, "\\p{Line_Break=Numeric}")); + rules.add(new RegexRule("$NU × ($AL | $HL)", "\\p{Line_Break=Numeric}", Resolution.NO_BREAK, "([\\p{Line_Break=Ambiguous} \\p{Line_Break=Alphabetic} \\p{Line_Break=Surrogate} \\p{Line_Break=Unknown} [\\p{Line_Break=Complex_Context}-\\p{gc=Mn}-\\p{gc=Mc}]] | \\p{Line_Break=HL})")); + rules.add(new RegexRule("$PR × ($ID | $EB | $EM)", "\\p{Line_Break=Prefix_Numeric}", Resolution.NO_BREAK, "(\\p{Line_Break=Ideographic} | \\p{Line_Break=E_Base} | \\p{Line_Break=E_Modifier})")); + rules.add(new RegexRule("($ID | $EB | $EM) × $PO", "(\\p{Line_Break=Ideographic} | \\p{Line_Break=E_Base} | \\p{Line_Break=E_Modifier})", Resolution.NO_BREAK, "\\p{Line_Break=Postfix_Numeric}")); + rules.add(new RegexRule("($PR | $PO) × ($AL | $HL)", "(\\p{Line_Break=Prefix_Numeric} | \\p{Line_Break=Postfix_Numeric})", Resolution.NO_BREAK, "([\\p{Line_Break=Ambiguous} \\p{Line_Break=Alphabetic} \\p{Line_Break=Surrogate} \\p{Line_Break=Unknown} [\\p{Line_Break=Complex_Context}-\\p{gc=Mn}-\\p{gc=Mc}]] | \\p{Line_Break=HL})")); + rules.add(new RegexRule("($AL | $HL) × ($PR | $PO)", "([\\p{Line_Break=Ambiguous} \\p{Line_Break=Alphabetic} \\p{Line_Break=Surrogate} \\p{Line_Break=Unknown} [\\p{Line_Break=Complex_Context}-\\p{gc=Mn}-\\p{gc=Mc}]] | \\p{Line_Break=HL})", Resolution.NO_BREAK, "(\\p{Line_Break=Prefix_Numeric} | \\p{Line_Break=Postfix_Numeric})")); + rules.add(new RegexRule("$NU ( $SY | $IS )* $CL × $PO", "\\p{Line_Break=Numeric} ( \\p{Line_Break=Break_Symbols} | \\p{Line_Break=Infix_Numeric} )* \\p{Line_Break=Close_Punctuation}", Resolution.NO_BREAK, "\\p{Line_Break=Postfix_Numeric}")); + rules.add(new RegexRule("$NU ( $SY | $IS )* $CP × $PO", "\\p{Line_Break=Numeric} ( \\p{Line_Break=Break_Symbols} | \\p{Line_Break=Infix_Numeric} )* \\p{Line_Break=CP}", Resolution.NO_BREAK, "\\p{Line_Break=Postfix_Numeric}")); + rules.add(new RegexRule("$NU ( $SY | $IS )* $CL × $PR", "\\p{Line_Break=Numeric} ( \\p{Line_Break=Break_Symbols} | \\p{Line_Break=Infix_Numeric} )* \\p{Line_Break=Close_Punctuation}", Resolution.NO_BREAK, "\\p{Line_Break=Prefix_Numeric}")); + rules.add(new RegexRule("$NU ( $SY | $IS )* $CP × $PR", "\\p{Line_Break=Numeric} ( \\p{Line_Break=Break_Symbols} | \\p{Line_Break=Infix_Numeric} )* \\p{Line_Break=CP}", Resolution.NO_BREAK, "\\p{Line_Break=Prefix_Numeric}")); + rules.add(new RegexRule("$NU ( $SY | $IS )* × $PO", "\\p{Line_Break=Numeric} ( \\p{Line_Break=Break_Symbols} | \\p{Line_Break=Infix_Numeric} )*", Resolution.NO_BREAK, "\\p{Line_Break=Postfix_Numeric}")); + rules.add(new RegexRule("$NU ( $SY | $IS )* × $PR", "\\p{Line_Break=Numeric} ( \\p{Line_Break=Break_Symbols} | \\p{Line_Break=Infix_Numeric} )*", Resolution.NO_BREAK, "\\p{Line_Break=Prefix_Numeric}")); + rules.add(new RegexRule("$PO × $OP $NU", "\\p{Line_Break=Postfix_Numeric}", Resolution.NO_BREAK, "\\p{Line_Break=Open_Punctuation} \\p{Line_Break=Numeric}")); + rules.add(new RegexRule("$PO × $OP $IS $NU", "\\p{Line_Break=Postfix_Numeric}", Resolution.NO_BREAK, "\\p{Line_Break=Open_Punctuation} \\p{Line_Break=Infix_Numeric} \\p{Line_Break=Numeric}")); + rules.add(new RegexRule("$PO × $NU", "\\p{Line_Break=Postfix_Numeric}", Resolution.NO_BREAK, "\\p{Line_Break=Numeric}")); + rules.add(new RegexRule("$PR × $OP $NU", "\\p{Line_Break=Prefix_Numeric}", Resolution.NO_BREAK, "\\p{Line_Break=Open_Punctuation} \\p{Line_Break=Numeric}")); + rules.add(new RegexRule("$PR × $OP $IS $NU", "\\p{Line_Break=Prefix_Numeric}", Resolution.NO_BREAK, "\\p{Line_Break=Open_Punctuation} \\p{Line_Break=Infix_Numeric} \\p{Line_Break=Numeric}")); + rules.add(new RegexRule("$PR × $NU", "\\p{Line_Break=Prefix_Numeric}", Resolution.NO_BREAK, "\\p{Line_Break=Numeric}")); + rules.add(new RegexRule("$HY × $NU", "\\p{Line_Break=Hyphen}", Resolution.NO_BREAK, "\\p{Line_Break=Numeric}")); + rules.add(new RegexRule("$IS × $NU", "\\p{Line_Break=Infix_Numeric}", Resolution.NO_BREAK, "\\p{Line_Break=Numeric}")); + rules.add(new RegexRule("$NU ( $SY | $IS )* × $NU", "\\p{Line_Break=Numeric} ( \\p{Line_Break=Break_Symbols} | \\p{Line_Break=Infix_Numeric} )*", Resolution.NO_BREAK, "\\p{Line_Break=Numeric}")); + rules.add(new RegexRule("$JL × $JL | $JV | $H2 | $H3", "\\p{Line_Break=JL}", Resolution.NO_BREAK, "\\p{Line_Break=JL} | \\p{Line_Break=JV} | \\p{Line_Break=H2} | \\p{Line_Break=H3}")); + rules.add(new RegexRule("$JV | $H2 × $JV | $JT", "\\p{Line_Break=JV} | \\p{Line_Break=H2}", Resolution.NO_BREAK, "\\p{Line_Break=JV} | \\p{Line_Break=JT}")); + rules.add(new RegexRule("$JT | $H3 × $JT", "\\p{Line_Break=JT} | \\p{Line_Break=H3}", Resolution.NO_BREAK, "\\p{Line_Break=JT}")); + rules.add(new RegexRule("$JL | $JV | $JT | $H2 | $H3 × $PO", "\\p{Line_Break=JL} | \\p{Line_Break=JV} | \\p{Line_Break=JT} | \\p{Line_Break=H2} | \\p{Line_Break=H3}", Resolution.NO_BREAK, "\\p{Line_Break=Postfix_Numeric}")); + rules.add(new RegexRule("$PR × $JL | $JV | $JT | $H2 | $H3", "\\p{Line_Break=Prefix_Numeric}", Resolution.NO_BREAK, "\\p{Line_Break=JL} | \\p{Line_Break=JV} | \\p{Line_Break=JT} | \\p{Line_Break=H2} | \\p{Line_Break=H3}")); + rules.add(new RegexRule("($AL | $HL) × ($AL | $HL)", "([\\p{Line_Break=Ambiguous} \\p{Line_Break=Alphabetic} \\p{Line_Break=Surrogate} \\p{Line_Break=Unknown} [\\p{Line_Break=Complex_Context}-\\p{gc=Mn}-\\p{gc=Mc}]] | \\p{Line_Break=HL})", Resolution.NO_BREAK, "([\\p{Line_Break=Ambiguous} \\p{Line_Break=Alphabetic} \\p{Line_Break=Surrogate} \\p{Line_Break=Unknown} [\\p{Line_Break=Complex_Context}-\\p{gc=Mn}-\\p{gc=Mc}]] | \\p{Line_Break=HL})")); + rules.add(new RegexRule("$AP × ($AK | $DottedCircle | $AS)", "\\p{Line_Break=Aksara_Prebase}", Resolution.NO_BREAK, "(\\p{Line_Break=Aksara} | [◌] | \\p{Line_Break=Aksara_Start})")); + rules.add(new RegexRule("($AK | $DottedCircle | $AS) × ($VF | $VI)", "(\\p{Line_Break=Aksara} | [◌] | \\p{Line_Break=Aksara_Start})", Resolution.NO_BREAK, "(\\p{Line_Break=Virama_Final} | \\p{Line_Break=Virama})")); + rules.add(new RegexRule("($AK | $DottedCircle | $AS) $VI × ($AK | $DottedCircle)", "(\\p{Line_Break=Aksara} | [◌] | \\p{Line_Break=Aksara_Start}) \\p{Line_Break=Virama}", Resolution.NO_BREAK, "(\\p{Line_Break=Aksara} | [◌])")); + rules.add(new RegexRule("($AK | $DottedCircle | $AS) × ($AK | $DottedCircle | $AS) $VF", "(\\p{Line_Break=Aksara} | [◌] | \\p{Line_Break=Aksara_Start})", Resolution.NO_BREAK, "(\\p{Line_Break=Aksara} | [◌] | \\p{Line_Break=Aksara_Start}) \\p{Line_Break=Virama_Final}")); + rules.add(new RegexRule("$IS × ($AL | $HL)", "\\p{Line_Break=Infix_Numeric}", Resolution.NO_BREAK, "([\\p{Line_Break=Ambiguous} \\p{Line_Break=Alphabetic} \\p{Line_Break=Surrogate} \\p{Line_Break=Unknown} [\\p{Line_Break=Complex_Context}-\\p{gc=Mn}-\\p{gc=Mc}]] | \\p{Line_Break=HL})")); + rules.add(new RegexRule("($AL | $HL | $NU) × $OPmEastAsian", "([\\p{Line_Break=Ambiguous} \\p{Line_Break=Alphabetic} \\p{Line_Break=Surrogate} \\p{Line_Break=Unknown} [\\p{Line_Break=Complex_Context}-\\p{gc=Mn}-\\p{gc=Mc}]] | \\p{Line_Break=HL} | \\p{Line_Break=Numeric})", Resolution.NO_BREAK, "[\\p{Line_Break=Open_Punctuation}-[\\p{ea=F}\\p{ea=W}\\p{ea=H}]]")); + rules.add(new RegexRule("$CPmEastAsian × ($AL | $HL | $NU)", "[\\p{Line_Break=CP}-[\\p{ea=F}\\p{ea=W}\\p{ea=H}]]", Resolution.NO_BREAK, "([\\p{Line_Break=Ambiguous} \\p{Line_Break=Alphabetic} \\p{Line_Break=Surrogate} \\p{Line_Break=Unknown} [\\p{Line_Break=Complex_Context}-\\p{gc=Mn}-\\p{gc=Mc}]] | \\p{Line_Break=HL} | \\p{Line_Break=Numeric})")); + rules.add(new RegexRule("$sot ($RI $RI)* $RI × $RI", "^ (\\p{Line_Break=Regional_Indicator} \\p{Line_Break=Regional_Indicator})* \\p{Line_Break=Regional_Indicator}", Resolution.NO_BREAK, "\\p{Line_Break=Regional_Indicator}")); + rules.add(new RegexRule("[^$RI] ($RI $RI)* $RI × $RI", "[^\\p{Line_Break=Regional_Indicator}] (\\p{Line_Break=Regional_Indicator} \\p{Line_Break=Regional_Indicator})* \\p{Line_Break=Regional_Indicator}", Resolution.NO_BREAK, "\\p{Line_Break=Regional_Indicator}")); + rules.add(new RegexRule("$RI ÷ $RI", "\\p{Line_Break=Regional_Indicator}", Resolution.BREAK, "\\p{Line_Break=Regional_Indicator}")); + rules.add(new RegexRule("$EB × $EM", "\\p{Line_Break=E_Base}", Resolution.NO_BREAK, "\\p{Line_Break=E_Modifier}")); + rules.add(new RegexRule("$ExtPictUnassigned × $EM", "[\\p{Extended_Pictographic}&\\p{gc=Cn}]", Resolution.NO_BREAK, "\\p{Line_Break=E_Modifier}")); + // --- End of generated code. --- + + // TODO(egg): This could just as well be part of the rules… + rules.add(new RegexRule("(ALL ÷ / ÷ ALL)", "", Resolution.BREAK, "")); + + final UnicodeSet lbSA = new UnicodeSet("\\p{lb=SA}"); + for (final NamedSet part : partition) { + if (lbSA.containsAll(part.set)) { continue; } - partition.add(new NamedSet(lbValueShortName, "\\p{lb=" + lbValueShortName + "}")); - } - for (final NamedSet refinement : interestingSets) { - for (int i = 0; i < partition.size();) { - final String name = partition.get(i).name; - final UnicodeSet set = partition.get(i).set; - final UnicodeSet intersection = new UnicodeSet(set).retainAll(refinement.set); - final UnicodeSet complement = new UnicodeSet(set).removeAll(refinement.set); - if (!intersection.isEmpty() && !complement.isEmpty()) { - partition.add(i, new NamedSet(name, complement)); - partition.add(i + 1, new NamedSet(name + "&" + refinement.name, intersection)); - partition.remove(i + 2); - i += 2; - } else { - ++i; - } - } - } - for (final NamedSet part : partition) { fSets.add(part.set); fClassNames.add(part.name); } @@ -890,877 +882,32 @@ class NamedSet { void setText(StringBuffer s) { fText = s; prepareAppliedRules(s.length()); - } - - - - - @Override - int next(int startPos) { - int pos; // Index of the char following a potential break position - int thisChar; // Character at above position "pos" - - int prevPos; // Index of the char preceding a potential break position - int prevChar; // Character at above position. Note that prevChar - // // and thisChar may not be adjacent because combining - // // characters between them will be ignored. - - int prevPosX2; - int prevCharX2; // Character before prevChar, more context for LB 21a - - int nextPos; // Index of the next character following pos. - // // Usually skips over combining marks. - int tPos; // temp value. - int matchVals[] = null; // Number Expression Match Results - - - if (startPos >= fText.length()) { - return -1; + StringBuilder remapped = new StringBuilder(s.toString()); + resolved = new BreakContext[s.length() + 1]; + for (int i = 0; i < resolved.length; ++i) { + resolved[i] = new BreakContext(i); } - - - // Initial values for loop. Loop will run the first time without finding breaks, - // while the invalid values shift out and the "this" and - // "prev" positions are filled in with good values. - pos = prevPos = prevPosX2 = -1; // Invalid value, serves as flag for initial loop iteration. - thisChar = prevChar = prevCharX2 = 0; - nextPos = startPos; - - - // Loop runs once per position in the test text, until a break position - // is found. In each iteration, we are testing for a possible break - // just preceding the character at index "pos". The character preceding - // this char is at position "prevPos"; because of combining sequences, - // "prevPos" can be arbitrarily far before "pos". - for (;;) { - // Advance to the next position to be tested. - prevPosX2 = prevPos; - prevCharX2 = prevChar; - prevPos = pos; - prevChar = thisChar; - pos = nextPos; - nextPos = moveIndex32(fText, pos, 1); - - if (pos >= fText.length()) { - setAppliedRule(pos, "LB 2 Break at end of text"); - break; - } - - // We do this rule out-of-order because the adjustment does - // not effect the way that rules LB 3 through LB 6 match, - // and doing it here rather than after LB 6 is substantially - // simpler when combining sequences do occur. - - - // LB 9 Keep combining sequences together. - // advance over any CM class chars at "pos", - // result is "nextPos" for the following loop iteration. - thisChar = UTF16.charAt(fText, pos); - if (!(fSP.contains(thisChar) || fBK.contains(thisChar) || thisChar==0x0d || - thisChar==0x0a || fNL.contains(thisChar) || fZW.contains(thisChar) )) { - for (;;) { - if (nextPos == fText.length()) { - break; - } - int nextChar = UTF16.charAt(fText, nextPos); - if (!fCM.contains(nextChar)) { - break; - } - nextPos = moveIndex32(fText, nextPos, 1); - } - } - - // LB 9 Treat X CM* as if it were X - // No explicit action required. - - // LB 10 Treat any remaining combining mark as lb=AL, ea=Na - if (fCM.contains(thisChar)) { - thisChar = 'A'; - } - - - // If the loop is still warming up - if we haven't shifted the initial - // -1 positions out of prevPos yet - loop back to advance the - // position in the input without any further looking for breaks. - if (prevPos == -1) { - setAppliedRule(pos, "LB 9 adjust for combining sequences."); - continue; - } - - if (fBK.contains(prevChar)) { - setAppliedRule(pos, "LB 4 Always break after hard line breaks"); - break; - } - - if (fCR.contains(prevChar) && fLF.contains(thisChar)) { - setAppliedRule(pos, "LB 5 Break after CR, LF, NL, but not inside CR LF"); - continue; - } - if (fCR.contains(prevChar) || - fLF.contains(prevChar) || - fNL.contains(prevChar)) { - setAppliedRule(pos, "LB 5 Break after CR, LF, NL, but not inside CR LF"); - break; - } - - if (fBK.contains(thisChar) || fCR.contains(thisChar) || - fLF.contains(thisChar) || fNL.contains(thisChar) ) { - setAppliedRule(pos, "LB 6 Don't break before hard line breaks"); - continue; - } - - - if (fSP.contains(thisChar)) { - setAppliedRule(pos, "LB 7 Don't break before spaces or zero-width space"); - continue; - } - - if (fZW.contains(thisChar)) { - setAppliedRule(pos, "LB 7 Don't break before spaces or zero-width space"); - continue; - } - - // ZW SP* ÷ - // Scan backwards from prevChar for SP* ZW - tPos = prevPos; - while (tPos > 0 && fSP.contains(UTF16.charAt(fText, tPos))) { - tPos = moveIndex32(fText, tPos, -1); - } - if (fZW.contains(UTF16.charAt(fText, tPos))) { - setAppliedRule(pos, "LB 8 Break after zero width space"); - break; - } - - // The monkey test's way of ignoring combining characters doesn't work - // for this rule. ZWJ is also a CM. Need to get the actual character - // preceding "thisChar", not ignoring combining marks, possibly ZWJ. - { - int prevC = fText.codePointBefore(pos); - if (fZWJ.contains(prevC)) { - setAppliedRule(pos, "LB 8a ZWJ x"); - continue; - } - } - - // appliedRule: "LB 9, 10"; // Already done, at top of loop."; - - - // x WJ - // WJ x - if (fWJ.contains(thisChar) || fWJ.contains(prevChar)) { - setAppliedRule(pos, "LB 11 Do not break before or after WORD JOINER and related characters."); - continue; - } - - - if (fGL.contains(prevChar)) { - setAppliedRule(pos, "LB 12 GL x"); - continue; - } - - if (!(fSP.contains(prevChar) || - fBA.contains(prevChar) || - fHY.contains(prevChar) ) && fGL.contains(thisChar)) { - setAppliedRule(pos, "LB 12a [^SP BA HY] x GL"); - continue; - } - - if (fCL.contains(thisChar) || - fCP.contains(thisChar) || - fEX.contains(thisChar) || - fSY.contains(thisChar)) { - setAppliedRule(pos, "LB 13 Don't break before closings"); - continue; - } - - // Scan backwards, checking for this sequence. - // The OP char could include combining marks, so we actually check for - // OP CM* SP* x - tPos = prevPos; - if (fSP.contains(prevChar)) { - while (tPos > 0 && fSP.contains(UTF16.charAt(fText, tPos))) { - tPos=moveIndex32(fText, tPos, -1); - } - } - while (tPos > 0 && fCM.contains(UTF16.charAt(fText, tPos))) { - tPos=moveIndex32(fText, tPos, -1); - } - if (fOP.contains(UTF16.charAt(fText, tPos))) { - setAppliedRule(pos, "LB 14 Don't break after OP SP*"); - continue; - } - - // Same as LB 14, scan backward for - // (sot | BK | CR | LF | NL | OP CM*| QU CM* | GL CM* | SP) [\p{Pi}&QU] CM* SP*. - tPos = prevPos; - // SP* (with the aforementioned Twist). - if (fSP.contains(prevChar)) { - while (tPos > 0 && fSP.contains(UTF16.charAt(fText, tPos))) { - tPos = moveIndex32(fText, tPos, -1); - } - } - // CM*. - while (tPos > 0 && fCM.contains(UTF16.charAt(fText, tPos))) { - tPos = moveIndex32(fText, tPos, -1); - } - // [\p{Pi}&QU]. - if (fPi.contains(UTF16.charAt(fText, tPos)) && fQU.contains(UTF16.charAt(fText, tPos))) { - if (tPos == 0) { - setAppliedRule(pos, "LB 15a sot [\\p{Pi}&QU] SP* ×"); - continue; - } else { - tPos = moveIndex32(fText, tPos, -1); - if (fBK.contains(UTF16.charAt(fText, tPos)) || fCR.contains(UTF16.charAt(fText, tPos)) || - fLF.contains(UTF16.charAt(fText, tPos)) || fNL.contains(UTF16.charAt(fText, tPos)) || - fSP.contains(UTF16.charAt(fText, tPos)) || fZW.contains(UTF16.charAt(fText, tPos))) { - setAppliedRule(pos, "LB 15a (BK | CR | LF | NL | SP | ZW) [\\p{Pi}&QU] SP* ×"); - continue; - } - } - // CM*. - while (tPos > 0 && fCM.contains(UTF16.charAt(fText, tPos))) { - tPos = moveIndex32(fText, tPos, -1); - } - if (fOP.contains(UTF16.charAt(fText, tPos)) || fQU.contains(UTF16.charAt(fText, tPos)) || - fGL.contains(UTF16.charAt(fText, tPos))) { - setAppliedRule(pos, "LB 15a (OP | QU | GL) [\\p{Pi}&QU] SP* ×"); - continue; - } - } - - if (fPf.contains(thisChar) && fQU.contains(thisChar)) { - int nextChar = (nextPos < fText.length())? UTF16.charAt(fText, nextPos): 0; - if (nextPos == fText.length() || fSP.contains(nextChar) || fGL.contains(nextChar) || - fWJ.contains(nextChar) || fCL.contains(nextChar) || fQU.contains(nextChar) || - fCP.contains(nextChar) || fEX.contains(nextChar) || fIS.contains(nextChar) || - fSY.contains(nextChar) || fBK.contains(nextChar) || fCR.contains(nextChar) || - fLF.contains(nextChar) || fNL.contains(nextChar) || fZW.contains(nextChar)) { - setAppliedRule(pos, "LB 15b × [\\p{Pf}&QU] ( SP | GL | WJ | CL | QU | CP | EX | IS | SY | BK | CR | LF | NL | ZW | eot)"); - continue; - } - } - - if (nextPos < fText.length()) { - int nextChar = fText.codePointAt(nextPos); - if (fSP.contains(prevChar) && fIS.contains(thisChar) && fNU.contains(nextChar)) { - setAppliedRule(pos, "LB 15c Break before an IS that begins a number and follows a space"); - break; - } - } - - if (fIS.contains(thisChar)) { - setAppliedRule(pos, "LB 15d Do not break before numeric separators, even after spaces"); - continue; - } - - if (fNS.contains(thisChar)) { - tPos = prevPos; - while (tPos > 0 && fSP.contains(UTF16.charAt(fText, tPos))) { - tPos = moveIndex32(fText, tPos, -1); - } - while (tPos > 0 && fCM.contains(UTF16.charAt(fText, tPos))) { - tPos = moveIndex32(fText, tPos, -1); - } - if (fCL.contains(UTF16.charAt(fText, tPos)) || fCP.contains(UTF16.charAt(fText, tPos))) { - setAppliedRule(pos, "LB 16 (CL | CP) SP* x NS"); - continue; - } - } - - - if (fB2.contains(thisChar)) { - tPos = prevPos; - while (tPos > 0 && fSP.contains(UTF16.charAt(fText, tPos))) { - tPos = moveIndex32(fText, tPos, -1); - } - while (tPos > 0 && fCM.contains(UTF16.charAt(fText, tPos))) { - tPos = moveIndex32(fText, tPos, -1); - } - if (fB2.contains(UTF16.charAt(fText, tPos))) { - setAppliedRule(pos, "LB 17 B2 SP* x B2"); - continue; - } - } - - if (fSP.contains(prevChar)) { - setAppliedRule(pos, "LB 18 break after space"); - break; - } - - // LB 19 - // × [QU-\p{Pi}] - if (fQU.contains(thisChar) && !fPi.contains(thisChar)) { - setAppliedRule(pos, "LB 19 × [QU-\\p{Pi}]"); - continue; - } - // [QU-\p{Pf}] × - if (fQU.contains(prevChar) && !fPf.contains(prevChar)) { - setAppliedRule(pos, "LB 19 [QU-\\p{Pf}] ×"); - continue; - } - - // LB 19a - // [^\p{ea=F}\p{ea=W}\p{ea=H}] × QU - if (!feaFWH.contains(prevChar) && fQU.contains(thisChar)) { - setAppliedRule(pos, "LB 19a [^\\p{ea=F}\\p{ea=W}\\p{ea=H}] × QU"); - continue; - } - // × QU ( [^\p{ea=F}\p{ea=W}\p{ea=H}] | eot ) - if (fQU.contains(thisChar)) { - if (nextPos < fText.length()) { - int nextChar = fText.codePointAt(nextPos); - if (!feaFWH.contains(nextChar)) { - setAppliedRule(pos, "LB 19a × QU [^\\p{ea=F}\\p{ea=W}\\p{ea=H}]"); - continue; - } - } else { - setAppliedRule(pos, "LB 19 × QU eot"); - continue; - } - } - // QU × [^\p{ea=F}\p{ea=W}\p{ea=H}] - if (fQU.contains(prevChar) && !feaFWH.contains(thisChar)) { - setAppliedRule(pos, "LB 19a QU × [^\\p{ea=F}\\p{ea=W}\\p{ea=H}]"); - continue; - } - // ( sot | [^\p{ea=F}\p{ea=W}\p{ea=H}] ) QU × - if (fQU.contains(prevChar)) { - if (prevPos == 0) { - setAppliedRule(pos, "LB 19a sot QU ×"); - continue; - } - // prevPosX2 is -1 if there was a break, and prevCharX2 is 0; but the UAX #14 rules can - // look through breaks. - int breakObliviousPrevPosX2 = moveIndex32(fText, prevPos, -1); - while (fCM.contains(fText.codePointAt(breakObliviousPrevPosX2))) { - if (breakObliviousPrevPosX2 == 0) { - break; - } - int beforeCM = moveIndex32(fText, breakObliviousPrevPosX2, -1); - if (fBK.contains(fText.codePointAt(beforeCM)) || - fCR.contains(fText.codePointAt(beforeCM)) || - fLF.contains(fText.codePointAt(beforeCM)) || - fNL.contains(fText.codePointAt(beforeCM)) || - fSP.contains(fText.codePointAt(beforeCM)) || - fZW.contains(fText.codePointAt(beforeCM))) { - break; - } - breakObliviousPrevPosX2 = beforeCM; - } - if (!feaFWH.contains(fText.codePointAt(breakObliviousPrevPosX2)) || - fCM.contains(fText.codePointAt(breakObliviousPrevPosX2))) { - setAppliedRule(pos, "LB 19a [^\\p{ea=F}\\p{ea=W}\\p{ea=H}] QU ×"); - continue; - } - } - - if (fCB.contains(thisChar) || fCB.contains(prevChar)) { - setAppliedRule(pos, "LB 20 Break around a CB"); - break; - } - - // Don't break between Hyphens and letters if a break or a space precedes the hyphen. - // Formerly this was a Finnish tailoring. - // (sot | BK | CR | LF | NL | SP | ZW | CB | GL) ( HY | [\u2010] ) × AL - if (fAL.contains(thisChar) && (fHY.contains(prevChar) || fHH.contains(prevChar))) { - // sot ( HY | [\u2010] ) × AL. - if (prevPos == 0) { - setAppliedRule(pos, "LB 20a"); - continue; - } - // prevPosX2 is -1 if there was a break; but the UAX #14 rules can - // look through breaks. - int breakObliviousPrevPosX2 = moveIndex32(fText, prevPos, -1); - if (fBK.contains(fText.codePointAt(breakObliviousPrevPosX2)) || - fCR.contains(fText.codePointAt(breakObliviousPrevPosX2)) || - fLF.contains(fText.codePointAt(breakObliviousPrevPosX2)) || - fNL.contains(fText.codePointAt(breakObliviousPrevPosX2)) || - fSP.contains(fText.codePointAt(breakObliviousPrevPosX2)) || - fZW.contains(fText.codePointAt(breakObliviousPrevPosX2))) { - setAppliedRule(pos, "LB 20a"); - continue; - } - while (breakObliviousPrevPosX2 > 0 && - fCM.contains(fText.codePointAt(breakObliviousPrevPosX2))) { - breakObliviousPrevPosX2 = moveIndex32(fText, breakObliviousPrevPosX2, -1); - } - if (fCB.contains(fText.codePointAt(breakObliviousPrevPosX2)) || - fGL.contains(fText.codePointAt(breakObliviousPrevPosX2))) { - setAppliedRule(pos, "LB 20a"); - continue; - } - } - - if (fBA.contains(thisChar) || - fHY.contains(thisChar) || - fNS.contains(thisChar) || - fBB.contains(prevChar) ) { - setAppliedRule(pos, "LB 21"); - continue; - } - - if (fHL.contains(prevCharX2) && - (fHY.contains(prevChar) || - (fBA.contains(prevChar) && !feaFWH.contains(prevChar))) && - !fHL.contains(thisChar)) { - setAppliedRule(pos, "LB 21a HL (HY | BA) x [^HL]"); - continue; - } - - if (fSY.contains(prevChar) && fHL.contains(thisChar)) { - setAppliedRule(pos, "LB 21b SY x HL"); - continue; - } - - if (fIN.contains(thisChar)) { - setAppliedRule(pos, "LB 22"); - continue; - } - - // (AL | HL) x NU - // NU x (AL | HL) - if ((fAL.contains(prevChar) || fHL.contains(prevChar)) && fNU.contains(thisChar)) { - setAppliedRule(pos, "LB 23"); - continue; - } - if (fNU.contains(prevChar) && (fAL.contains(thisChar) || fHL.contains(thisChar))) { - setAppliedRule(pos, "LB 23"); - continue; - } - - // Do not break between numeric prefixes and ideographs, or between ideographs and numeric postfixes. - // PR x (ID | EB | EM) - // (ID | EB | EM) x PO - if (fPR.contains(prevChar) && - (fID.contains(thisChar) || fEB.contains(thisChar) || fEM.contains(thisChar))) { - setAppliedRule(pos, "LB 23a"); - continue; - } - if ((fID.contains(prevChar) || fEB.contains(prevChar) || fEM.contains(prevChar)) && - fPO.contains(thisChar)) { - setAppliedRule(pos, "LB 23a"); - continue; - } - - // Do not break between prefix and letters or ideographs. - // (PR | PO) x (AL | HL) - // (AL | HL) x (PR | PO) - if ((fPR.contains(prevChar) || fPO.contains(prevChar)) && - (fAL.contains(thisChar) || fHL.contains(thisChar))) { - setAppliedRule(pos, "LB 24 no break between prefix and letters or ideographs"); - continue; - } - if ((fAL.contains(prevChar) || fHL.contains(prevChar)) && - (fPR.contains(thisChar) || fPO.contains(thisChar))) { - setAppliedRule(pos, "LB 24 no break between prefix and letters or ideographs"); - continue; - } - - boolean continueToNextPosition = false; - // LB 25. - for (XUnicodeSet[] pair : new XUnicodeSet[][]{ - new XUnicodeSet[]{fCL, fPO}, // 1. NU (SY | IS)* CL × PO - new XUnicodeSet[]{fCP, fPO}, // 2. NU (SY | IS)* CP × PO - new XUnicodeSet[]{fCL, fPR}, // 3. NU (SY | IS)* CL × PR - new XUnicodeSet[]{fCP, fPR}, // 4. NU (SY | IS)* CP × PR - }) { - XUnicodeSet left = pair[0]; - XUnicodeSet right = pair[1]; - if (left.contains(prevChar) && right.contains(thisChar)) { - // Check for the NU (SY | IS)* part. - boolean leftHandSideMatches = false; - tPos = moveIndex32(fText, prevPos, -1); - for (;;) { - while (tPos > 0 && fCM.contains(fText.codePointAt(tPos))) { - tPos = moveIndex32(fText, tPos, -1); - } - final int tChar = fText.codePointAt(tPos); - if (fSY.contains(tChar) || fIS.contains(tChar)) { - if (tPos == 0) { - leftHandSideMatches = false; - break; - } - tPos = moveIndex32(fText, tPos, -1); - } else if (fNU.contains(tChar)) { - leftHandSideMatches = true; - break; - } else { - leftHandSideMatches = false; - break; - } - } - if (leftHandSideMatches) { - setAppliedRule(pos, "LB 25/1..4"); - continueToNextPosition = true; - break; - } - } - } - if (continueToNextPosition) { - continue; - } - // 5. NU (SY | IS)* × PO - // 6. NU (SY | IS)* × PR - // 13. NU (SY | IS)* × NU - boolean leftHandSideMatches; - tPos = prevPos; - for (;;) { - while (tPos > 0 && fCM.contains(fText.codePointAt(tPos))) { - tPos = moveIndex32(fText, tPos, -1); - } - final int tChar = fText.codePointAt(tPos); - if (fSY.contains(tChar) || fIS.contains(tChar)) { - if (tPos == 0) { - leftHandSideMatches = false; - break; - } - tPos = moveIndex32(fText, tPos, -1); - } else if (fNU.contains(tChar)) { - leftHandSideMatches = true; - break; - } else { - leftHandSideMatches = false; - break; - } - } - if (leftHandSideMatches && - (fPO.contains(thisChar) || fPR.contains(thisChar) || fNU.contains(thisChar))) { - setAppliedRule(pos, "LB 25/5,6,13,14"); - continue; - } - if (nextPos < fText.length()) { - final int nextChar = fText.codePointAt(nextPos); - // 7. PO × OP NU - if (fPO.contains(prevChar) && fOP.contains(thisChar) && fNU.contains(nextChar)) { - setAppliedRule(pos, "LB 25/7"); - continue; - } - // 9. PR × OP NU - if (fPR.contains(prevChar) && fOP.contains(thisChar) && fNU.contains(nextChar)) { - setAppliedRule(pos, "LB 25/9"); - continue; - } - int nextPosX2 = moveIndex32(fText, nextPos, 1); - while (nextPosX2 < fText.length() && fCM.contains(fText.codePointAt(nextPosX2))) { - nextPosX2 = moveIndex32(fText, nextPosX2, 1); - } - - if (nextPosX2 < fText.length()) { - final int nextCharX2 = fText.codePointAt(nextPosX2); - // 7bis. PO × OP IS NU - if (fPO.contains(prevChar) && fOP.contains(thisChar) && fIS.contains(nextChar) && - fNU.contains(nextCharX2)) { - setAppliedRule(pos, "LB 25/7bis"); - continue; - } - // 9bis. PR × OP IS NU - if (fPR.contains(prevChar) && fOP.contains(thisChar) && fIS.contains(nextChar) && - fNU.contains(nextCharX2)) { - setAppliedRule(pos, "LB 25/9bis"); - continue; - } - } - } - for (XUnicodeSet[] pair : new XUnicodeSet[][]{ - new XUnicodeSet[]{fPO, fNU}, // 8. PO × NU - new XUnicodeSet[]{fPR, fNU}, // 10. PR × NU - new XUnicodeSet[]{fHY, fNU}, // 11. HY × NU - new XUnicodeSet[]{fIS, fNU}, // 12. IS × NU - }) { - XUnicodeSet left = pair[0]; - XUnicodeSet right = pair[1]; - if (left.contains(prevChar) && right.contains(thisChar)) { - continueToNextPosition = true; - break; - } - } - if (continueToNextPosition) { - continue; - } - - if (fJL.contains(prevChar) && (fJL.contains(thisChar) || - fJV.contains(thisChar) || - fH2.contains(thisChar) || - fH3.contains(thisChar))) { - setAppliedRule(pos, "LB 26 Do not break a Korean syllable."); - continue; - } - - if ((fJV.contains(prevChar) || fH2.contains(prevChar)) && - (fJV.contains(thisChar) || fJT.contains(thisChar))) { - setAppliedRule(pos, "LB 26 Do not break a Korean syllable."); - continue; - } - - if ((fJT.contains(prevChar) || fH3.contains(prevChar)) && - fJT.contains(thisChar)) { - setAppliedRule(pos, "LB 26 Do not break a Korean syllable."); - continue; - } - - if ((fJL.contains(prevChar) || fJV.contains(prevChar) || - fJT.contains(prevChar) || fH2.contains(prevChar) || fH3.contains(prevChar)) && - fPO.contains(thisChar)) { - setAppliedRule(pos, "LB 27 Treat a Korean Syllable Block the same as ID."); - continue; - } - if (fPR.contains(prevChar) && (fJL.contains(thisChar) || fJV.contains(thisChar) || - fJT.contains(thisChar) || fH2.contains(thisChar) || fH3.contains(thisChar))) { - setAppliedRule(pos, "LB 27 Treat a Korean Syllable Block the same as ID."); - continue; - } - - - - if ((fAL.contains(prevChar) || fHL.contains(prevChar)) && (fAL.contains(thisChar) || fHL.contains(thisChar))) { - setAppliedRule(pos, "LB 28 Do not break between alphabetics"); - continue; - } - - if (fAP.contains(prevChar) && - (fAK.contains(thisChar) || thisChar == '◌' || fAS.contains(thisChar))) { - setAppliedRule(pos, "LB 28a.1 AP x (AK | ◌ | AS)"); - continue; - } - - if ((fAK.contains(prevChar) || prevChar == '◌' || fAS.contains(prevChar)) && - (fVF.contains(thisChar) || fVI.contains(thisChar))) { - setAppliedRule(pos, "LB 28a.2 (AK | ◌ | AS) x (VF | VI)"); - continue; - } - - if ((fAK.contains(prevCharX2) || prevCharX2 == '◌' || fAS.contains(prevCharX2)) && - fVI.contains(prevChar) && - (fAK.contains(thisChar) || thisChar == '◌')) { - setAppliedRule(pos, "LB 28a.3 (AK | ◌ | AS) VI x (AK | ◌)"); - continue; - } - - if (nextPos < fText.length()) { - // note: UnicodeString::char32At(length) returns ffff, not distinguishable - // from a legit ffff noncharacter. So test length separately. - int nextChar = UTF16.charAt(fText, nextPos); - if ((fAK.contains(prevChar) || prevChar == '◌' || fAS.contains(prevChar)) && - (fAK.contains(thisChar) || thisChar == '◌' || fAS.contains(thisChar)) && - fVF.contains(nextChar)) { - setAppliedRule(pos, "LB 28a.4 (AK | ◌ | AS) x (AK | ◌ | AS) VF"); - continue; - } - } - - if (fIS.contains(prevChar) && (fAL.contains(thisChar) || fHL.contains(thisChar))) { - setAppliedRule(pos, "LB 29 Do not break between numeric punctuation and alphabetics"); - continue; - } - - // (AL | NU) x OP - // CP x (AL | NU) - if ((fAL.contains(prevChar) || fHL.contains(prevChar) || fNU.contains(prevChar)) && - fOP30.contains(thisChar)) { - setAppliedRule(pos, "LB 30 Do not break between letters, numbers, or ordinary symbols and opening or closing punctuation."); - continue; - } - if (fCP30.contains(prevChar) && - (fAL.contains(thisChar) || fHL.contains(thisChar) || fNU.contains(thisChar))) { - setAppliedRule(pos, "LB 30 Do not break between letters, numbers, or ordinary symbols and opening or closing punctuation."); - continue; - } - - // RI RI ÷ RI - // RI x RI - if (fRI.contains(prevCharX2) && fRI.contains(prevChar) && fRI.contains(thisChar)) { - setAppliedRule(pos, "LB 30a Break between pairs of Regional Indicators."); - break; - } - if (fRI.contains(prevChar) && fRI.contains(thisChar)) { - // Two Regional Indicators have been paired. - // Over-write the trailing one (thisChar) to prevent it from forming another pair with a - // following RI. This is a hack. - thisChar = -1; - setAppliedRule(pos, "LB 30a Break between pairs of Regional Indicators."); - continue; - } - - // LB30b Do not break between an emoji base (or potential emoji) and an emoji modifier. - if (fEB.contains(prevChar) && fEM.contains(thisChar)) { - setAppliedRule(pos, "LB 30b Emoji Base x Emoji Modifier"); - continue; - } - - if (fExtPictUnassigned.contains(prevChar) && fEM.contains(thisChar)) { - setAppliedRule(pos, "LB30b [\\p{Extended_Pictographic}&\\p{Cn}] × EM"); - continue; + for (final SegmentationRule rule : rules) { + rule.apply(remapped, resolved); + } + for (int i = 0; i < resolved.length; ++i) { + if (resolved[i].appliedRule == null) { + throw new IllegalArgumentException("Failed to resolve at " + i); } - - // LB 31 Break everywhere else - setAppliedRule(pos, "LB 31 Break everywhere else"); - break; + setAppliedRule(i, resolved[i].appliedRule.name()); } - - return pos; } - - - // Match the following regular expression in the input text. - // ((PR | PO) CM*)? ((OP | HY) CM*)? (IS CM*)? NU CM* ((NU | IS | SY) CM*) * ((CL | CP) CM*)? (PR | PO) CM*)? - // 0 0 1 4 4 4 5 5 7 7 7 7 9 9 9 11 11 (match states) - // retVals array [0] index of the start of the match, or -1 if no match - // [1] index of first char following the match. - // Can not use Java regex because need supplementary character support, - // and because Unicode char properties version must be the same as in - // the version of ICU being tested. - private int[] LBNumberCheck(StringBuffer s, int startIdx, int[] retVals) { - if (retVals == null) { - retVals = new int[2]; - } - retVals[0] = -1; // Indicates no match. - int matchState = 0; - int idx = startIdx; - - matchLoop: for (idx = startIdx; idx= 7) { - retVals[0] = startIdx; - retVals[1] = idx; - } - return retVals; + return -1; } - @Override List charClasses() { return fSets; @@ -2136,6 +1283,9 @@ static int nextCP(StringBuffer s, int i) { } + + + /** * random number generator. Not using Java's built-in Randoms for two reasons: * 1. Using this code allows obtaining the same sequences as those from the ICU4C monkey test. @@ -2151,6 +1301,9 @@ private static int m_rand() return (m_seed >>> 16) % 32768; } + private final static String[] monkeys = new String[] { + "🙈", "🙉", "🙊", "🐵", "🐒"}; + // Helper function for formatting error output. // Append a string into a fixed-size field in a StringBuffer. // Blank-pad the string if it is shorter than the field. @@ -2214,6 +1367,7 @@ void RunMonkey(BreakIterator bi, RBBIMonkeyKind mk, String name, int seed, int boolean[] precedingBreaks = new boolean[TESTSTRINGLEN*2 + 1]; int i; int loopCount = 0; + int errorCount = 0; boolean printTestData = false; boolean printBreaksFromBI = false; @@ -2253,16 +1407,13 @@ void RunMonkey(BreakIterator bi, RBBIMonkeyKind mk, String name, int seed, int // For minimizing width of class name output. int classNameSize = mk.maxClassNameSize(); - - int dotsOnLine = 0; while (loopCount < numIterations || numIterations == -1) { if (numIterations == -1 && loopCount % 10 == 0) { // If test is running in an infinite loop, display a periodic tic so // we can tell that it is making progress. - System.out.print("."); - if (dotsOnLine++ >= 80){ - System.out.println(); - dotsOnLine = 0; + System.out.print(monkeys[m_rand() % monkeys.length]); + if (loopCount % 1_000_000 == 0) { + System.out.println("\nTested " + loopCount / 1_000_000 + " million random strings with " + errorCount + " errors"); } } // Save current random number seed, so that we can recreate the random numbers @@ -2274,6 +1425,7 @@ void RunMonkey(BreakIterator bi, RBBIMonkeyKind mk, String name, int seed, int if (printTestData) { System.out.println("Test Data string ..."); } + final boolean java8OrOlder = System.getProperty("java.version").startsWith("1."); for (i=0; i 0 && @@ -2423,6 +1582,7 @@ void RunMonkey(BreakIterator bi, RBBIMonkeyKind mk, String name, int seed, int } if (errorType != null) { + ++errorCount; // Format a range of the test text that includes the failure as // a data item that can be included in the rbbi test data file. @@ -2465,12 +1625,14 @@ void RunMonkey(BreakIterator bi, RBBIMonkeyKind mk, String name, int seed, int buffer.append("\n") .append((expectedBreaks[i] ? "Break expected but not found." : "Break found but not expected.")) .append( - String.format(" at index %d. Parameters to reproduce: @\"type=%s seed=%d loop=1\"\n", + String.format(" at index %d. Parameters to reproduce: -Dtest=RBBITestMonkey#Test%sMonkey -Dseed=%d -Dloop=1\n", i, name, seed)); int c; // Char from test data for (ci = startContext; ci <= endContext && ci != -1; ci = nextCP(testText, ci)) { - + if (ci == testText.length()) { + break; // TODO(egg): The index dance above seems wrong. + } c = testText.codePointAt(ci); buffer.append((ci == i) ? " --→" : " ") .append(String.format(" %3d : ", ci)) @@ -2514,7 +1676,7 @@ public void TestCharMonkey() { RBBICharMonkey m = new RBBICharMonkey(); BreakIterator bi = BreakIterator.getCharacterInstance(Locale.US); - RunMonkey(bi, m, "char", seed, loopCount); + RunMonkey(bi, m, "Char", seed, loopCount); } @Test @@ -2525,7 +1687,7 @@ public void TestWordMonkey() { logln("Word Break Monkey Test"); RBBIWordMonkey m = new RBBIWordMonkey(); BreakIterator bi = BreakIterator.getWordInstance(Locale.US); - RunMonkey(bi, m, "word", seed, loopCount); + RunMonkey(bi, m, "Word", seed, loopCount); } @Test @@ -2537,7 +1699,7 @@ public void TestLineMonkey() { RBBILineMonkey m = new RBBILineMonkey(); BreakIterator bi = BreakIterator.getLineInstance(Locale.US); try { - RunMonkey(bi, m, "line", seed, loopCount); + RunMonkey(bi, m, "Line", seed, loopCount); } catch(IllegalArgumentException e) { if (e.getMessage().equals("Invalid code point U+-000001")) { // Looks like you used class UnicodeSet instead of class XUnicodeSet @@ -2558,7 +1720,7 @@ public void TestSentMonkey() { logln("Sentence Break Monkey Test"); RBBISentenceMonkey m = new RBBISentenceMonkey(); BreakIterator bi = BreakIterator.getSentenceInstance(Locale.US); - RunMonkey(bi, m, "sent", seed, loopCount); + RunMonkey(bi, m, "Sent", seed, loopCount); } // // Round-trip monkey tests. @@ -2579,7 +1741,7 @@ public void TestRTCharMonkey() { BreakIterator bi = BreakIterator.getCharacterInstance(Locale.US); String rules = bi.toString(); BreakIterator rtbi = new RuleBasedBreakIterator(rules); - RunMonkey(rtbi, m, "char", seed, loopCount); + RunMonkey(rtbi, m, "RTChar", seed, loopCount); } @Test @@ -2592,7 +1754,7 @@ public void TestRTWordMonkey() { BreakIterator bi = BreakIterator.getWordInstance(Locale.US); String rules = bi.toString(); BreakIterator rtbi = new RuleBasedBreakIterator(rules); - RunMonkey(rtbi, m, "word", seed, loopCount); + RunMonkey(rtbi, m, "RTWord", seed, loopCount); } @Test @@ -2606,7 +1768,7 @@ public void TestRTLineMonkey() { String rules = bi.toString(); BreakIterator rtbi = new RuleBasedBreakIterator(rules); try { - RunMonkey(rtbi, m, "line", seed, loopCount); + RunMonkey(rtbi, m, "RTLine", seed, loopCount); } catch(IllegalArgumentException e) { if (e.getMessage().equals("Invalid code point U+-000001")) { // Looks like you used class UnicodeSet instead of class XUnicodeSet @@ -2629,6 +1791,6 @@ public void TestRTSentMonkey() { BreakIterator bi = BreakIterator.getSentenceInstance(Locale.US); String rules = bi.toString(); BreakIterator rtbi = new RuleBasedBreakIterator(rules); - RunMonkey(rtbi, m, "sent", seed, loopCount); + RunMonkey(rtbi, m, "RTSent", seed, loopCount); } } diff --git a/icu4j/main/core/src/test/java/com/ibm/icu/dev/test/rbbi/RegexRule.java b/icu4j/main/core/src/test/java/com/ibm/icu/dev/test/rbbi/RegexRule.java new file mode 100644 index 000000000000..8695f0f5ac37 --- /dev/null +++ b/icu4j/main/core/src/test/java/com/ibm/icu/dev/test/rbbi/RegexRule.java @@ -0,0 +1,111 @@ +// © 2024 and later: Unicode, Inc. and others. +// License & terms of use: http://www.unicode.org/copyright.html + +package com.ibm.icu.dev.test.rbbi; + +import java.util.Arrays; +import java.util.Optional; +import java.util.regex.Matcher; +import java.util.regex.Pattern; + +/** + * A regex rule expressed as in UAXes #14 and #29. + * + * The rule consists of two regexes for context before and after a position in + * the remapped text, + * and of a resolution (break or not) that applies to the corresponding position + * in the original + * string if both match. + */ +class RegexRule extends SegmentationRule { + RegexRule(String name, String before, Resolution resolution, + String after) { + super(name); + resolution_ = resolution; + before_ = Pattern.compile(expandUnicodeSets(before), Pattern.COMMENTS | Pattern.DOTALL); + endsWithBefore_ = Pattern.compile( + ".*(" + expandUnicodeSets(before) + ")", Pattern.COMMENTS | Pattern.DOTALL); + after_ = Pattern.compile(expandUnicodeSets(after), Pattern.COMMENTS | Pattern.DOTALL); + } + + @Override + void apply(StringBuilder remapped, BreakContext[] resolved) { + // The unicodetools implementation simply tries, for each index, to + // match the string up to the index against /.*(before)/ (with + // `matches`) and the beginning of the string after the index against + // /after/ (with `lookingAt`), but that is very slow, especially for + // nonempty /before/. While the old monkeys are not a production + // implementation, we still do not want them to be too slow, since we + // need to test millions of sample strings. Instead we search for + // /before/ and /after/, and check resulting candidates. This speeds + // things up by a factor of ~40. + // We need to be careful about greedy matching: The first position where + // the rule matches may be before the end of the first /before/ match. + // However, it is both: + // 1. within a /before/ match or at its bounds, + // 2. at the beginning of an /after/ match. + // Further, the /before/ context of the rule matches within the + // aforementioned /before/ match. Note that we need to look for + // overlapping matches, thus calls to `find` are always preceded by a + // reset via `region`. + final Matcher beforeSearch = before_.matcher(remapped); + final Matcher afterSearch = after_.matcher(remapped); + beforeSearch.useAnchoringBounds(false); + afterSearch.useAnchoringBounds(false); + if (beforeSearch.find() && afterSearch.find()) { + for (;;) { + if (afterSearch.start() < beforeSearch.start()) { + afterSearch.region(beforeSearch.start(), remapped.length()); + if (!afterSearch.find()) { + break; + } + } else if (afterSearch.start() > beforeSearch.end()) { + if (beforeSearch.start() == remapped.length()) { + break; + } + beforeSearch.region(remapped.offsetByCodePoints(beforeSearch.start(), 1), + remapped.length()); + if (!beforeSearch.find()) { + break; + } + } else { + final Optional position = Arrays.stream(resolved) + .filter(r -> r.indexInRemapped != null && r.indexInRemapped == afterSearch.start()) + .findFirst(); + if (!position.isPresent()) { + throw new IllegalArgumentException(("Rule " + name() + + " matched at position " + afterSearch.start() + + " in " + remapped + + " which does not correspond to an index in " + + "the original string")); + } + if (position.get().appliedRule == null && + endsWithBefore_.matcher(remapped) + .useAnchoringBounds(false) + .region(beforeSearch.start(), afterSearch.start()) + .matches()) { + position.get().appliedRule = this; + } + if (afterSearch.start() == remapped.length()) { + break; + } + afterSearch.region(remapped.offsetByCodePoints(afterSearch.start(), 1), + remapped.length()); + if (!afterSearch.find()) { + break; + } + } + } + } + } + + @Override + Resolution resolution() { + return resolution_; + } + + private final Pattern before_; + private final Pattern endsWithBefore_; + private final Pattern after_; + private final Resolution resolution_; +} \ No newline at end of file diff --git a/icu4j/main/core/src/test/java/com/ibm/icu/dev/test/rbbi/RemapRule.java b/icu4j/main/core/src/test/java/com/ibm/icu/dev/test/rbbi/RemapRule.java new file mode 100644 index 000000000000..6c84f6052b93 --- /dev/null +++ b/icu4j/main/core/src/test/java/com/ibm/icu/dev/test/rbbi/RemapRule.java @@ -0,0 +1,166 @@ +// © 2024 and later: Unicode, Inc. and others. +// License & terms of use: http://www.unicode.org/copyright.html + +package com.ibm.icu.dev.test.rbbi; + +import java.util.regex.Matcher; +import java.util.regex.Pattern; + +/** + * A segmentation rule expressed as in UAXes #14 and #29. + * + * The application of a remap rule is a normal regex replacement on the remapped + * string. This replacement may use capturing groups. Any positions in the + * original string that correspond to positions within the replaced text are + * resolved to NO_BREAK by this rule. + */ +public class RemapRule extends SegmentationRule { + RemapRule(String name, String pattern, String replacement) { + super(name); + replacement_ = replacement; + pattern_ = Pattern.compile(expandUnicodeSets(pattern), Pattern.COMMENTS | Pattern.DOTALL); + } + + @Override + void apply(StringBuilder remapped, BreakContext[] resolved) { + // This one has to be a StringBuffer rather than a StringBuilder because the + // overload of + // AppendReplacement that takes a StringBuilder is new in Java 9. + StringBuffer result = new StringBuffer(); + int i = 0; + int offset = 0; + // We find all matches of the `pattern_` and replace them according to + // the `replacement_`, producing the new remapped string `result`. + // For every position i in the original string, + // `resolved[i].indexInRemapped` is null if i lies within a replaced + // match, and is set to the new index in `result` otherwise, by adding + // the accumulated difference `offset` between match lengths and + // replacement lengths. + // Consider a 4-codepoint, 6 code unit string s = ⟨ 𒀀, ◌́, ␠, ◌𝅲 ⟩, where + // ␠ stands for U+0020 and U+12000 𒀀 and U+1D172 ◌𝅲 each require two code + // units, and apply the following two rules: + // 1. (?\P{lb=SP}) \p{lb=CM}* → ${X} + // 2. \p{lb=CM} → A + // The string remapped and the indexInRemapped values change as follows: + // indexInRemapped remapped string rule final + // (aligned on the initial string) applied offset + // 𒀀 ◌́ ␠ ◌𝅲 + // 0 1 2 3 4 5 6 ⟨ 𒀀, ◌́, ␠, ◌𝅲 ⟩ (none) + // 0 - - 2 3 4 5 ⟨ 𒀀, ␠, ◌𝅲 ⟩ 1 -1 + // 0 - - 2 3 - 4 ⟨ 𒀀, ␠, A ⟩ 2 -1 + // + // Note that the last indexInRemapped is always equal to the length of + // the remapped string. + final Matcher matcher = pattern_.matcher(remapped); + while (matcher.find()) { + for (;; ++i) { + if (resolved[i].indexInRemapped == null) { + continue; + } + if (resolved[i].indexInRemapped != null && + resolved[i].indexInRemapped > matcher.start()) { + break; + } + resolved[i].indexInRemapped += offset; + } + for (;; ++i) { + if (resolved[i].indexInRemapped == null) { + continue; + } + // Note that + // `*resolved[i].indexInRemapped > matcher.end()` should + // never happen with ordinary rules, but could in principle + // happen with rules that remap to code point sequences, e.g., + // 1. BC → TYZ + // 2. AT → X + // applied to ⟨ A, B, C ⟩: + // indexInRemapped remapped rule + // A B C + // 0 1 2 3 ⟨ A, B, C ⟩ (none) + // 0 1 - 4 ⟨ A, T, Y, Z ⟩ 1 + // 0 - - 3 ⟨ X, Y, Z ⟩ 2 + // Where for the application of rule 2, the match ends at + // position 2 in remapped, which does not correspond to a + // position in the original string. + if (resolved[i].indexInRemapped != null && + resolved[i].indexInRemapped >= matcher.end()) { + break; + } + if (resolved[i].appliedRule != null && + resolved[i].appliedRule.resolution() == Resolution.BREAK) { + throw new IllegalArgumentException( + "Replacement rule at remapped indices " + + matcher.start() + + " sqq. spans a break"); + } + resolved[i].appliedRule = this; + resolved[i].indexInRemapped = null; + } + // While replacing, we need to check that we are not creating + // surrogate pairs. Since appendReplacement performs two + // concatenations (the unreplaced segment and the replacement), we + // need to check in two places: whether the unreplaced segment + // starts with a trailing surrogate that ends up after a leading + // surrogate, and whether the replaced segment starts with a leading + // surrogate that ends up after a trailing surrogate. + // We break the pair by replacing one of the surrogates with U+FFFF, + // which has the same properties for all but line breaking, and the + // same behaviour in line breaking (lb=SG and lb=XX are both treated + // as lb=AL). + Integer trailingLead = null; + if (result.length() > 0 && Character.isHighSurrogate(result.charAt(result.length() - 1))) { + trailingLead = result.length() - 1; + } + + matcher.appendReplacement(result, replacement_); + + if (trailingLead != null && trailingLead + 1 < result.length() && + Character.isLowSurrogate(result.charAt(trailingLead + 1))) { + result.setCharAt(trailingLead, '\uFFFF'); + } + + if (matcher.start() + offset > 0 && + Character.isHighSurrogate(result.charAt(matcher.start() + offset - 1)) && + Character.isLowSurrogate(result.charAt(matcher.start() + offset))) { + result.setCharAt(matcher.start() + offset, '\uFFFF'); + } + offset = result.length() - resolved[i].indexInRemapped; + } + for (; i < resolved.length; ++i) { + if (resolved[i].indexInRemapped == null) { + continue; + } + resolved[i].indexInRemapped += offset; + } + + Integer trailingLead = null; + if (result.length() > 0 && Character.isHighSurrogate(result.charAt(result.length() - 1))) { + trailingLead = result.length() - 1; + } + matcher.appendTail(result); + if (trailingLead != null && trailingLead + 1 < result.length() && + Character.isLowSurrogate(result.charAt(trailingLead + 1))) { + result.setCharAt(trailingLead, '\uFFFF'); + } + + if (resolved[resolved.length - 1].indexInRemapped != result.length()) { + StringBuilder indices = new StringBuilder(); + for (final BreakContext r : resolved) { + indices.append(r.indexInRemapped == null ? "null" : r.indexInRemapped.toString()); + indices.append(","); + } + throw new IllegalArgumentException("Inconsistent indexInRemapped " + indices + " for new remapped string " + + result); + } + remapped.setLength(0); + remapped.append(result); + } + + @Override + Resolution resolution() { + return Resolution.NO_BREAK; + } + + private final Pattern pattern_; + private final String replacement_; +} \ No newline at end of file diff --git a/icu4j/main/core/src/test/java/com/ibm/icu/dev/test/rbbi/SegmentationRule.java b/icu4j/main/core/src/test/java/com/ibm/icu/dev/test/rbbi/SegmentationRule.java new file mode 100644 index 000000000000..b9507d0ce3d1 --- /dev/null +++ b/icu4j/main/core/src/test/java/com/ibm/icu/dev/test/rbbi/SegmentationRule.java @@ -0,0 +1,94 @@ +// © 2024 and later: Unicode, Inc. and others. +// License & terms of use: http://www.unicode.org/copyright.html + +package com.ibm.icu.dev.test.rbbi; + +import java.text.ParsePosition; + +import javax.swing.RowFilter.Entry; + +import com.ibm.icu.impl.Utility; +import com.ibm.icu.text.UnicodeSet; +import com.ibm.icu.text.UnicodeSet.EntryRange; +import com.ibm.icu.text.UTF16; + +/** + * A segmentation rule expressed as in UAXes #14 and #29. + * + * Rules are applied sequentially. + * Rules operate on a mutable remapped string (which the caller should initially + * set to the string + * to be segmented), and can resolve positions in the original string to either + * BREAK or NO_BREAK. + */ +public abstract class SegmentationRule { + enum Resolution { + BREAK, + NO_BREAK, + } + + static class BreakContext { + BreakContext(int index) { + indexInRemapped = index; + } + + Integer indexInRemapped; + SegmentationRule appliedRule = null; + }; + + SegmentationRule(String name) { + name_ = name; + } + + // Returns "\\uhhhh" for a BMP code point and "\\uDhhh\\uDhhh" (UTF-16) for other code points. + private String javaUEscape(int codePoint) { + if (codePoint <= 0xFFFF) { + return "\\u" + Utility.hex(codePoint); + } else { + return "\\u" + Utility.hex(UTF16.getLeadSurrogate(codePoint)) + "\\u" + + Utility.hex(UTF16.getTrailSurrogate(codePoint)); + } + } + + protected String expandUnicodeSets(String regex) { + StringBuilder result = new StringBuilder(); + int i = 0; + final boolean java8OrOlder = System.getProperty("java.version").startsWith("1."); + while (i < regex.length()) { + if (regex.charAt(i) == '[' || regex.charAt(i) == '\\') { + ParsePosition pp = new ParsePosition(i); + final UnicodeSet set = new UnicodeSet(regex, pp, null); + // Regular expressions that match unpaired surrogates apparently behave + // differently in Java 8. Let’s not go there. + if (java8OrOlder) { + set.removeAll(new UnicodeSet("[\\uD800-\\uDFFF]")); + } + // Escape everything. We could use _generatePattern, but then we would have to + // convert \U escapes to sequences of \‌u escapes, and to escape # ourselves. + result.append('['); + for (EntryRange range : set.ranges()) { + result.append(javaUEscape(range.codepoint)); + if (range.codepointEnd != range.codepoint) { + result.append('-'); + result.append(javaUEscape(range.codepointEnd)); + } + } + result.append(']'); + i = pp.getIndex(); + } else { + result.append(regex.charAt(i++)); + } + } + return result.toString(); + } + + abstract void apply(StringBuilder remapped, BreakContext[] resolved); + + abstract Resolution resolution(); + + String name() { + return name_; + } + + private final String name_; +}