/** ******************************************************************************* * Copyright (C) 1996-2001, International Business Machines Corporation and * * others. All Rights Reserved. * ******************************************************************************* * * $Source: /xsrl/Nsvn/icu/unicodetools/com/ibm/text/UCD/UCD.java,v $ * $Date: 2003/07/21 15:50:06 $ * $Revision: 1.28 $ * ******************************************************************************* */ package com.ibm.text.UCD; import java.util.List; import java.util.ArrayList; import java.util.HashMap; import java.util.BitSet; import java.util.Map; import java.io.IOException; import java.io.DataInputStream; import java.io.BufferedInputStream; import java.io.FileInputStream; import java.io.BufferedReader; import com.ibm.text.utility.*; import com.ibm.icu.text.UnicodeSet; public final class UCD implements UCD_Types { static final boolean DEBUG = false; /** * Used for the default version. */ public static final String latestVersion = "4.0.1"; /** * Create singleton instance for default (latest) version */ public static UCD make() { return make(""); } /** * Create singleton instance for the specific version */ public static UCD make(String version) { if (version == null || version.length() == 0) version = latestVersion; if (version.indexOf('.') < 0) throw new IllegalArgumentException("Version must be of form 3.1.1"); UCD result = (UCD)versionCache.get(version); if (result == null) { //System.out.println(Utility.getStack()); result = new UCD(); result.fillFromFile(version); versionCache.put(version, result); } return result; } /** * Get the version of the UCD */ public String getVersion() { return version; } /** * Get the date that the data was parsed */ public long getDate() { return date; } /** * Is the code point allocated? */ public boolean isAllocated(int codePoint) { if (getCategory(codePoint) != Cn) return true; if (compositeVersion >= 0x20000 && codePoint >= 0xF0000 && codePoint <= 0x10FFFD) return true; if (isNoncharacter(codePoint)) return true; return false; } public boolean isNoncharacter(int codePoint) { if ((codePoint & 0xFFFE) == 0xFFFE) { if (compositeVersion < 0x20000 && codePoint > 0xFFFF) return false; // major < 2 return true; } if (codePoint >= 0xFDD0 && codePoint <= 0xFDEF && compositeVersion >= 0x30100) return true; // major >= 3 && minor >= 1 return false; } /** * Is the code point assigned to a character (or surrogate) */ public boolean isAssigned(int codePoint) { return getCategory(codePoint) != Cn; } /** * Is the code point a PUA character (fast check) */ public boolean isPUA(int codePoint) { return (codePoint >= 0xE000 && codePoint < 0xF900 || codePoint >= 0xF0000 && codePoint < 0xFFFFE || codePoint >= 0x100000 && codePoint < 0x10FFFE); } /** * Many ranges are elided in the UCD. All but the first are not actually * represented in the data internally. This detects such cases. */ public boolean isRepresented(int codePoint) { return getRaw(codePoint) != null; } /** * Return XML version of the data associated with the code point. */ public String toString(int codePoint) { return get(codePoint, true).toString(this,FULL); } /** * Get the character name. */ public String getName(int codePoint) { return getName(codePoint, NORMAL); } /** * Get the character name. */ public String getName(String s) { return getName(s, NORMAL); } /** * Get the character name. */ public String getName(int codePoint, byte style) { if (style == SHORT) return get(codePoint, true).shortName; return get(codePoint, true).name; } /** * Get the character names for the code points in a string, separated by ", " */ public String getName(String s, byte style) { if (s.length() == 1) return getName(s.charAt(0), style); StringBuffer result = new StringBuffer(); int cp; for (int i = 0; i < s.length(); i += UTF32.count16(cp)) { cp = UTF32.char32At(s, i); if (i > 0) result.append(", "); result.append(getName(cp, style)); } return result.toString(); } /** * Get the code in U+ notation */ public static String getCode(int codePoint) { return "U+" + Utility.hex(codePoint); } /** * Get the code in U+ notation */ public static String getCode(String s) { if (s.length() == 1) return getCode(s.charAt(0)); // fast path StringBuffer result = new StringBuffer(); int cp; for (int i = 0; i < s.length(); i += UTF32.count16(cp)) { cp = UTF32.char32At(s, i); if (i > 0) result.append(", "); result.append(getCode(cp)); } return result.toString(); } /** * Get the name and number (U+xxxx NAME) for a code point */ public String getCodeAndName(int codePoint, byte type) { return getCode(codePoint) + " " + getName(codePoint, type); } /** * Get the name and number (U+xxxx NAME) for the code points in a string, * separated by ", " */ public String getCodeAndName(String s, byte type) { if (s == null || s.length() == 0) return "NULL"; if (s.length() == 1) return getCodeAndName(s.charAt(0)); // fast path StringBuffer result = new StringBuffer(); int cp; for (int i = 0; i < s.length(); i += UTF32.count16(cp)) { cp = UTF32.char32At(s, i); if (i > 0) result.append(", "); result.append(getCodeAndName(cp)); } return result.toString(); } /** * Get the name and number (U+xxxx NAME) for a code point */ public String getCodeAndName(int codePoint) { return getCodeAndName(codePoint, NORMAL); } /** * Get the name and number (U+xxxx NAME) for a code point */ public String getCodeAndName(String s) { return getCodeAndName(s, NORMAL); } /** * Get the general category */ public byte getCategory(int codePoint) { return get(codePoint, false).generalCategory; } private static final byte FAKE_SYMBOL = 57; // fake category for comparison private static final byte FAKE_PUNCTUATION = 58; // fake category for comparison private static final byte FAKE_SEPERATOR = 59; // fake category for comparison private static final byte FAKE_NUMBER = 60; // fake category for comparison private static final byte FAKE_MARK = 61; // fake category for comparison private static final byte FAKE_LETTER = 62; // fake category for comparison private static final byte FAKE_OTHER = 63; // fake category for comparison private static final byte FAKENC = 31; // fake category for comparison public byte getModCat(int cp, int collapseBits) { byte cat = getCategory(cp); if (cat == UNASSIGNED && isNoncharacter(cp)) { cat = FAKENC; } else if (((1< field in the UCD data. */ public byte getDecompositionType(int codePoint) { return get(codePoint, false).decompositionType; } IntMap hanExceptions = null; static class HanException { double numericValue; byte numericType; } void populateHanExceptions() { hanExceptions = new IntMap(); BufferedReader in = null; try { in = Utility.openUnicodeFile("Unihan", Default.ucdVersion, true, Utility.UTF8); int lineCounter = 0; while (true) { Utility.dot(++lineCounter); String line = in.readLine(); if (line == null) break; if (line.length() < 6) continue; if (line.charAt(0) == '#') continue; line = line.trim(); int tabPos = line.indexOf('\t'); int tabPos2 = line.indexOf('\t', tabPos+1); String property = line.substring(tabPos+1, tabPos2).trim(); if (!property.endsWith("Numeric")) continue; String propertyValue = line.substring(tabPos2+1).trim(); propertyValue = Utility.replace(propertyValue, ",", ""); int hack = propertyValue.indexOf(' '); if (hack >= 0) { Utility.fixDot(); System.out.println("BAD NUMBER: " + line); propertyValue = propertyValue.substring(0,hack); } String scode = line.substring(2, tabPos).trim(); int code = Integer.parseInt(scode, 16); if (code == 0x5793 || code == 0x4EAC) continue; // two exceptions!! //kAccountingNumeric //kOtherNumeric //kPrimaryNumeric HanException except = (HanException) hanExceptions.get(code); if (except != null) throw new Exception("Duplicate Numeric Value for " + line); except = new HanException(); hanExceptions.put(code, except); except.numericValue = Double.parseDouble(propertyValue); except.numericType = property.equals("kAccountingNumeric") ? NUMERIC : property.equals("kOtherNumeric") ? NUMERIC : property.equals("kPrimaryNumeric") ? NUMERIC : NONE; if (except.numericType == NONE) throw new Exception("Unknown Numeric Type for " + line); if (false) { Utility.fixDot(); System.out.println(line); System.out.println(getNumericValue(code)); System.out.println(getNumericTypeID(code)); } } in.close(); } catch (Exception e) { throw new ChainException("Han File Processing Exception", null, e); } finally { Utility.fixDot(); System.out.println("****Size: " + hanExceptions.size()); } } public double getNumericValue(int codePoint) { if (hanExceptions == null) populateHanExceptions(); Object except = hanExceptions.get(codePoint); if (except != null) { return ((HanException)except).numericValue; } return get(codePoint, false).numericValue; } public byte getNumericType(int codePoint) { if (hanExceptions == null) populateHanExceptions(); Object except = hanExceptions.get(codePoint); if (except != null) { return ((HanException)except).numericType; } return get(codePoint, false).numericType; } public String getCase(int codePoint, byte simpleVsFull, byte caseType) { return getCase(codePoint, simpleVsFull, caseType, ""); } public String getCase(String s, byte simpleVsFull, byte caseType) { return getCase(s, simpleVsFull, caseType, ""); } public String getCase(int codePoint, byte simpleVsFull, byte caseType, String condition) { UData udata = get(codePoint, true); if (caseType < LOWER || caseType > FOLD || (simpleVsFull != SIMPLE && simpleVsFull != FULL)) { throw new IllegalArgumentException("simpleVsFull or caseType out of bounds"); } if (caseType < FOLD) { if (simpleVsFull == FULL && udata.specialCasing.length() != 0) { if (condition.length() == 0 || udata.specialCasing.indexOf(condition) < 0) { simpleVsFull = SIMPLE; } } } else { // special case. For these characters alone, use "I" as option meaning collapse to "i" //if (codePoint == 0x0131 || codePoint == 0x0130) { // special case turkish i if (getBinaryProperty(codePoint, CaseFoldTurkishI)) { if (!udata.specialCasing.equals("I")) simpleVsFull = SIMPLE; else simpleVsFull = FULL; } } switch (caseType + simpleVsFull) { case SIMPLE + UPPER: return udata.simpleUppercase; case SIMPLE + LOWER: return udata.simpleLowercase; case SIMPLE + TITLE: return udata.simpleTitlecase; case SIMPLE + FOLD: return udata.simpleCaseFolding; case FULL + UPPER: return udata.fullUppercase; case FULL + LOWER: return udata.fullLowercase; case FULL + TITLE: return udata.fullTitlecase; case FULL + FOLD: return udata.fullCaseFolding; } throw new IllegalArgumentException("getCase: " + caseType + ", " + simpleVsFull); } static final char SHY = '\u00AD'; static final char APOSTROPHE = '\u2019'; public String getCase(String s, byte simpleVsFull, byte caseType, String condition) { if (UTF32.length32(s) == 1) return getCase(UTF32.char32At(s, 0), simpleVsFull, caseType); StringBuffer result = new StringBuffer(); int cp; byte currentCaseType = caseType; UnicodeProperty defaultIgnorable = DerivedProperty.make(DerivedProperty.DefaultIgnorable, this); for (int i = 0; i < s.length(); i += UTF32.count16(cp)) { cp = UTF32.char32At(s, i); String mappedVersion = getCase(cp, simpleVsFull, currentCaseType, condition); result.append(mappedVersion); if (caseType == TITLE) { // set the case type for the next character // certain characters are ignored if (cp == SHY || cp == '\'' || cp == APOSTROPHE) continue; byte cat = getCategory(cp); if (cat == Mn || cat == Me || cat == Cf || cat == Lm) continue; if (defaultIgnorable.hasValue(cp)) continue; // if DefaultIgnorable is not supported, then // check for (Cf + Cc + Cs) - White_Space // if (cat == Cs && cp != 0x85 && (cp < 9 || cp > 0xD)) continue; // if letter is cased, change next to lowercase, otherwise revert to TITLE if (cat == Lu || cat == Ll || cat == Lt || getBinaryProperty(cp, Other_Lowercase) // skip if not supported || getBinaryProperty(cp, Other_Uppercase) // skip if not supported ) { currentCaseType = LOWER; } else { currentCaseType = TITLE; } } } return result.toString(); } /* public String getSimpleLowercase(int codePoint) { return get(codePoint, true).simpleLowercase; } public String getSimpleUppercase(int codePoint) { return get(codePoint, true).simpleUppercase; } public String getSimpleTitlecase(int codePoint) { return get(codePoint, true).simpleTitlecase; } public String getSimpleCaseFolding(int codePoint) { return get(codePoint, true).simpleCaseFolding; } public String getFullLowercase(int codePoint) { return get(codePoint, true).fullLowercase; } public String getFullUppercase(int codePoint) { return get(codePoint, true).fullUppercase; } public String getFullTitlecase(int codePoint) { return get(codePoint, true).fullTitlecase; } public String getFullCaseFolding(int codePoint) { return get(codePoint, true).simpleCaseFolding; } public String getLowercase(int codePoint, boolean full) { if (full) return getFullLowercase(codePoint); return getSimpleLowercase(codePoint); } public String getUppercase(int codePoint, boolean full) { if (full) return getFullUppercase(codePoint); return getSimpleLowercase(codePoint); } public String getTitlecase(int codePoint, boolean full) { if (full) return getFullTitlecase(codePoint); return getSimpleTitlecase(codePoint); } public String getCaseFolding(int codePoint, boolean full) { if (full) return getFullCaseFolding(codePoint); return getSimpleCaseFolding(codePoint); } public String getLowercase(String s, boolean full) { if (s.length() == 1) return getLowercase(s.charAt(0), true); StringBuffer result = new StringBuffer(); int cp; for (int i = 0; i < s.length(); i += UTF32.count16(cp)) { cp = UTF32.char32At(s, i); if (i > 0) result.append(", "); result.append(getLowercase(cp, true)); } return result.toString(); } public String getUppercase(String s, boolean full) { if (s.length() == 1) return getUppercase(s.charAt(0), true); StringBuffer result = new StringBuffer(); int cp; for (int i = 0; i < s.length(); i += UTF32.count16(cp)) { cp = UTF32.char32At(s, i); if (i > 0) result.append(", "); result.append(getUppercase(cp, true)); } return result.toString(); } public String getTitlecase(String s, boolean full) { if (s.length() == 1) return getTitlecase(s.charAt(0), true); StringBuffer result = new StringBuffer(); int cp; for (int i = 0; i < s.length(); i += UTF32.count16(cp)) { cp = UTF32.char32At(s, i); if (i > 0) result.append(", "); result.append(getTitlecase(cp, true)); } return result.toString(); } public String getCaseFolding(String s, boolean full) { if (s.length() == 1) return getCaseFolding(s.charAt(0), true); StringBuffer result = new StringBuffer(); int cp; for (int i = 0; i < s.length(); i += UTF32.count16(cp)) { cp = UTF32.char32At(s, i); if (i > 0) result.append(", "); result.append(getCaseFolding(cp, true)); } return result.toString(); } */ public String getSpecialCase(int codePoint) { return get(codePoint, true).specialCasing; } public byte getEastAsianWidth(int codePoint) { // if (0x30000 <= codepoint && codepoint <= 0x3FFFD) return EAW; return get(codePoint, false).eastAsianWidth; } public byte getLineBreak(int codePoint) { return get(codePoint, false).lineBreak; } public byte getScript(int codePoint) { return get(codePoint, false).script; } public byte getScript(String s) { byte result = COMMON_SCRIPT; if (s == null || s.length() == 0) return result; int cp; for (int i = 0; i < s.length(); i += UTF32.count16(cp)) { cp = UTF32.char32At(s, i); byte script = getScript(cp); if (script == INHERITED_SCRIPT) continue; result = script; } return result; } public byte getAge(int codePoint) { return get(codePoint, false).age; } public byte getJoiningType(int codePoint) { return get(codePoint, false).joiningType; } public byte getJoiningGroup(int codePoint) { return get(codePoint, false).joiningGroup; } public long getBinaryProperties(int codePoint) { return get(codePoint, false).binaryProperties; } public boolean getBinaryProperty(int codePoint, int bit) { return (get(codePoint, false).binaryProperties & (1L<= 0xFC5E && cp <= 0xFC63 || cp == 0xFDFA || cp == 0xFDFB) return false; if (cp >= 0xFE70 && cp <= 0xFE7E && (cp & 1) == 0) return false; } */ byte cat = getCategory(cp); if (cat == Lu || cat == Ll || cat == Lt || cat == Lm || cat == Lo || cat == Nl) return true; if (getBinaryProperty(cp, Other_ID_Start)) return true; return false; } public boolean isIdentifierContinue_NO_Cf(int cp) { if (isIdentifierStart(cp)) return true; /* if (extended) { if (cp == 0x00B7) return true; if (cp == 0x0E33 || cp == 0x0EB3 || cp == 0xFF9E || cp == 0xFF9F) return true; } */ byte cat = getCategory(cp); if (cat == Mn || cat == Mc || cat == Nd || cat == Pc) return true; if (getBinaryProperty(cp, Other_ID_Start)) return true; return false; } public boolean isIdentifier(String s) { if (s.length() == 0) return false; // at least one! int cp; for (int i = 0; i < s.length(); i += UTF32.count16(cp)) { cp = UTF32.char32At(s, i); if (i == 0) { if (!isIdentifierStart(cp)) return false; } else { if (!isIdentifierContinue_NO_Cf(cp)) return false; } } return true; } /* Middle Dot. Because most Catalan legacy data will be encoded in Latin-1, U+00B7 MIDDLE DOT needs to be allowed in . In particular, the following four characters should be in and not : 0E33 THAI CHARACTER SARA AM 0EB3 LAO VOWEL SIGN AM FF9E HALFWIDTH KATAKANA VOICED SOUND MARK FF9F HALFWIDTH KATAKANA SEMI-VOICED SOUND MARK Irregularly decomposing characters. U+037A GREEK YPOGEGRAMMENI and certain Arabic presentation forms have irregular compatibility decompositions, and need to be excluded from both and . It is recommended that all Arabic presentation forms be excluded from identifiers in any event, although only a few of them are required to be excluded for normalization to guarantee identifier closure. */ // ******************* // PRIVATES // ******************* // cache of singletons private static Map versionCache = new HashMap(); private static final int LIMIT_CODE_POINT = 0x110000; private static final UData[] ALL_NULLS = new UData[1024]; // main data private UData[][] data = new UData[LIMIT_CODE_POINT>>10][]; // extras private BitSet combiningClassSet = new BitSet(256); private String version; private String file; private long date = -1; private byte format = -1; //private byte major = -1; //private byte minor = -1; //private byte update = -1; private int compositeVersion = -1; private int size = -1; // cache last UData private int lastCode = Integer.MIN_VALUE; private UData lastResult = UData.UNASSIGNED; private boolean lastCodeFixed = false; // hide constructor private UCD() { for (int i = 0; i < data.length; ++i) { data[i] = ALL_NULLS; } } private void add(UData uData) { int high = uData.codePoint>>10; if (data[high] == ALL_NULLS) { UData[] temp = new UData[1024]; data[high] = temp; } data[high][uData.codePoint & 0x3FF] = uData; } public boolean hasComputableName(int codePoint) { if (codePoint >= 0xF900 && codePoint <= 0xFA2D) return true; if (codePoint >= 0x2800 && codePoint <= 0x28FF) return true; if (codePoint >= 0x2F800 && codePoint <= 0x2FA1D) return true; int rangeStart = mapToRepresentative(codePoint, compositeVersion < 0x020105); switch (rangeStart) { default: return getRaw(codePoint) == null; case 0x2800: // braille case 0xF900: // compat ideos case 0x2F800: // compat ideos case 0x3400: // CJK Ideograph Extension A case 0x4E00: // CJK Ideograph case 0x20000: // Extension B case 0xAC00: // Hangul Syllable case 0xE000: // Private Use case 0xF0000: // Private Use case 0x100000: // Private Use case 0xD800: // Surrogate case 0xDB80: // Private Use case 0xDC00: // Private Use case 0xFFFF: // Noncharacter return true; } } private UData getRaw(int codePoint) { return data[codePoint>>10][codePoint & 0x3FF]; } // access data for codepoint UData get(int codePoint, boolean fixStrings) { /*if (codePoint == 0xF901) { System.out.println(version + ", " + Integer.toString(compositeVersion, 16)); System.out.println("debug: "); } */ if (codePoint < 0 || codePoint > 0x10FFFF) { throw new IllegalArgumentException("Illegal Code Point: " + Utility.hex(codePoint)); } //if (codePoint == lastCode && fixStrings <= lastCodeFixed) return lastResult; /* // we play some funny tricks for performance // if cp is not represented, it is either in a elided block or missing. // elided blocks are either CONTINUE or FFFF byte cat; if (!ucdData.isRepresented(cp)) { int rep = UCD.mapToRepresentative(cp); if (rep == 0xFFFF) cat = Cn; else if (rep != cp) return CONTINUE; else if (!ucdData.isRepresented(rep)) cat = Cn; else cat = ucdData.getCategory(rep); } else { cat = ucdData.getCategory(cp); } */ UData result = null; // do range stuff String constructedName = null; int rangeStart = mapToRepresentative(codePoint, compositeVersion < 0x020105); boolean isHangul = false; switch (rangeStart) { case 0xF900: if (compositeVersion < 0x020105) { if (fixStrings) constructedName = "CJK COMPATIBILITY IDEOGRAPH-" + Utility.hex(codePoint, 4); break; } // FALL THROUGH!!!! default: result = getRaw(codePoint); if (result == null) { result = UData.UNASSIGNED; result.name = null; // clean this up, since we reuse UNASSIGNED result.shortName = null; if (fixStrings) { result.name = ""; } } if (fixStrings) { if (result.name == null) { result.name = ""; // System.out.println("Warning: fixing name for " + result.name); } if (result.shortName == null) { result.shortName = Utility.replace(result.name, UCD_Names.NAME_ABBREVIATIONS); } } return result; case 0x3400: // CJK Ideograph Extension A case 0x4E00: // CJK Ideograph case 0x20000: // Extension B if (fixStrings) constructedName = "CJK UNIFIED IDEOGRAPH-" + Utility.hex(codePoint, 4); break; case 0xAC00: // Hangul Syllable isHangul = true; if (fixStrings) { constructedName = "HANGUL SYLLABLE " + getHangulName(codePoint); } break; case 0xE000: // Private Use case 0xF0000: // Private Use case 0x100000: // Private Use if (fixStrings) constructedName = ""; break; case 0xD800: // Surrogate case 0xDB80: // Private Use case 0xDC00: // Private Use if (fixStrings) constructedName = ""; break; case 0xFFFF: // Noncharacter if (fixStrings) constructedName = ""; break; } result = getRaw(rangeStart); if (result == null) { result = UData.UNASSIGNED; result.name = null; // clean this up, since we reuse UNASSIGNED result.shortName = null; if (fixStrings) { result.name = ""; result.shortName = Utility.replace(result.name, UCD_Names.NAME_ABBREVIATIONS); } return result; } result.codePoint = codePoint; if (fixStrings) { result.name = constructedName; result.shortName = Utility.replace(constructedName, UCD_Names.NAME_ABBREVIATIONS); result.decompositionMapping = result.bidiMirror = result.simpleLowercase = result.simpleUppercase = result.simpleTitlecase = result.simpleCaseFolding = result.fullLowercase = result.fullUppercase = result.fullTitlecase = result.fullCaseFolding = UTF32.valueOf32(codePoint); } if (isHangul) { if (fixStrings) result.decompositionMapping = getHangulDecompositionPair(codePoint); result.decompositionType = CANONICAL; } return result; } // Neither Mapped nor Composite CJK: [\u3400-\u4DB5\u4E00-\u9FA5\U00020000-\U0002A6D6] public static final boolean isCJK_AB(int bigChar) { return (CJK_A_BASE <= bigChar && bigChar < CJK_A_LIMIT || CJK_B_BASE <= bigChar && bigChar < CJK_B_LIMIT); } public static boolean isCJK_BASE(int cp) { return (CJK_BASE <= cp && cp < CJK_LIMIT || cp == 0xFA0E // compat characters that don't decompose. || cp == 0xFA0F || cp == 0xFA11 || cp == 0xFA13 || cp == 0xFA14 || cp == 0xFA1F || cp == 0xFA21 || cp == 0xFA23 || cp == 0xFA24 || cp == 0xFA27 || cp == 0xFA28 || cp == 0xFA29 || cp == 0xFA2E || cp == 0xFA2F ); } // Hangul constants public static final int SBase = 0xAC00, LBase = 0x1100, VBase = 0x1161, TBase = 0x11A7, LCount = 19, VCount = 21, TCount = 28, NCount = VCount * TCount, // 588 SCount = LCount * NCount, // 11172 LLimit = LBase + LCount, // 1113 VLimit = VBase + VCount, // 1176 TLimit = TBase + TCount, // 11C3 LLimitFull = 0x1160, VLimitFull = TBase, TLimitFull = 0x11FF, SLimit = SBase + SCount; // D7A4 private static String getHangulName(int s) { int SIndex = s - SBase; if (0 > SIndex || SIndex >= SCount) { throw new IllegalArgumentException("Not a Hangul Syllable: " + s); } int LIndex = SIndex / NCount; int VIndex = (SIndex % NCount) / TCount; int TIndex = SIndex % TCount; // if (true) return "?"; return UCD_Names.JAMO_L_TABLE[LIndex] + UCD_Names.JAMO_V_TABLE[VIndex] + UCD_Names.JAMO_T_TABLE[TIndex]; } private static final char[] pair = new char[2]; static boolean isDoubleHangul(int s) { int SIndex = s - SBase; if (0 > SIndex || SIndex >= SCount) { throw new IllegalArgumentException("Not a Hangul Syllable: " + s); } return (SIndex % TCount) == 0; } static String getHangulDecompositionPair(int ch) { int SIndex = ch - SBase; if (0 > SIndex || SIndex >= SCount) { return ""; } int TIndex = SIndex % TCount; if (TIndex != 0) { // triple pair[0] = (char)(SBase + SIndex - TIndex); pair[1] = (char)(TBase + TIndex); } else { pair[0] = (char)(LBase + SIndex / NCount); pair[1] = (char)(VBase + (SIndex % NCount) / TCount); } return String.valueOf(pair); } static int composeHangul(int char1, int char2) { if (LBase <= char1 && char1 < LLimit && VBase <= char2 && char2 < VLimit) { return (SBase + ((char1 - LBase) * VCount + (char2 - VBase)) * TCount); } if (SBase <= char1 && char1 < SLimit && TBase <= char2 && char2 < TLimit && ((char1 - SBase) % TCount) == 0) { return char1 + (char2 - TBase); } return 0xFFFF; // no composition } static public boolean isHangulSyllable(int char1) { return SBase <= char1 && char1 < SLimit; } static boolean isLeadingJamoComposition(int char1) { return isLeadingJamo(char1) || isLV(char1); } static boolean isLV(int char1) { return (SBase <= char1 && char1 < SLimit && ((char1 - SBase) % TCount) == 0); } static boolean isVowelJamo(int cp) { return (VBase <= cp && cp < VLimit); } static boolean isTrailingJamo(int cp) { return (TBase <= cp && cp < TLimit); } static boolean isLeadingJamo(int cp) { return (LBase <= cp && cp < LLimit); } static boolean isNonLeadJamo(int cp) { return (VBase <= cp && cp < VLimit) || (TBase <= cp && cp < TLimit); } byte getHangulSyllableType(int cp) { if (!isAssigned(cp)) return NA; if (LBase <= cp && cp < LLimitFull) return L; if (LLimitFull <= cp && cp < VLimitFull) return V; if (VLimitFull <= cp && cp < TLimitFull) return T; if (isLV(cp)) return LV; if (isHangulSyllable(cp)) return LVT; return NA; } static String getHangulSyllableTypeID_fromIndex(byte index, byte style) { if (style == LONG) return UCD_Names.LONG_HANGUL_SYLLABLE_TYPE[index]; return UCD_Names.HANGUL_SYLLABLE_TYPE[index]; } String getHangulSyllableTypeID(int char1, byte style) { return getHangulSyllableTypeID_fromIndex(getHangulSyllableType(char1),style); } private void fillFromFile(String version) { try { fillFromFile2(version); } catch (ChainException e) { try { ConvertUCD.main(new String[]{version}); } catch (Exception e2) { throw new ChainException("Can't build data file for {0}", new Object[]{version}, e2); } fillFromFile2(version); } } private void fillFromFile2(String version) { DataInputStream dataIn = null; String fileName = BIN_DIR + "UCD_Data" + version + ".bin"; int uDataFileCount = 0; try { dataIn = new DataInputStream( new BufferedInputStream( new FileInputStream(fileName), 128*1024)); // header format = dataIn.readByte(); byte major = dataIn.readByte(); byte minor = dataIn.readByte(); byte update = dataIn.readByte(); compositeVersion = (major << 16) | (minor << 8) | update; String foundVersion = major + "." + minor + "." + update; if (format != BINARY_FORMAT || !version.equals(foundVersion)) { throw new ChainException("Illegal data file format for {0}: {1}, {2}", new Object[]{version, new Byte(format), foundVersion}); } date = dataIn.readLong(); size = uDataFileCount = dataIn.readInt(); boolean didJoiningHack = false; System.out.println("Loading UCD " + foundVersion); // records for (int i = 0; i < uDataFileCount; ++i) { UData uData = new UData(); uData.readBytes(dataIn); if (uData.codePoint == 0x5E) { System.out.println("SPOT-CHECK: " + uData); } //T = Mc + (Cf - ZWNJ - ZWJ) int cp = uData.codePoint; byte old = uData.joiningType; byte cat = uData.generalCategory; //if (cp == 0x200D) { // uData.joiningType = JT_C; //} else /* if (cp != 0x200D && cp != 0x200C && (cat == Mn || cat == Cf)) { uData.joiningType = JT_T; } */ if (!didJoiningHack && uData.joiningType != old) { System.out.println("HACK " + foundVersion + ": Setting " + UCD_Names.LONG_JOINING_TYPE[uData.joiningType] + ": " + Utility.hex(cp) + " " + uData.name); didJoiningHack = true; } combiningClassSet.set(uData.combiningClass & 0xFF); add(uData); } /* if (update == -1) { throw new ChainException("Data File truncated for ", new Object[]{version}, e); } if (size != fileSize) { throw new ChainException("Counts do not match: file {0}, records {1}", new Object[]{new Integer(fileSize), new Integer(size)}); } */ // everything is ok! this.version = version; this.file = fileName; //+ " " + new File(fileName).lastModified(); } catch (IOException e) { throw new ChainException("Can't read data file for {0}", new Object[]{version}, e); } finally { if (dataIn != null) { try { dataIn.close(); } catch (IOException e) {} } } } public static class BlockData { public int start; public int end; public String name; } public boolean getBlockData(int blockId, BlockData output) { if (blocks == null) loadBlocks(); BlockData temp; try { temp = (BlockData) blocks.get(blockId); } catch (IndexOutOfBoundsException e) { return false; } output.name = temp.name; output.start = temp.start; output.end = temp.end; return true; } private List blocks = null; private void loadBlocks() { blocks = new ArrayList(); try { BufferedReader in = Utility.openUnicodeFile("Blocks", version, true, Utility.LATIN1); try { while (true) { // 0000..007F; Basic Latin String line = Utility.readDataLine(in); if (line == null) break; if (line.length() == 0) continue; int pos1 = line.indexOf('.'); int pos2 = line.indexOf(';', pos1); BlockData blockData = new BlockData(); blockData.start = Integer.parseInt(line.substring(0, pos1), 16); blockData.end = Integer.parseInt(line.substring(pos1+2, pos2), 16); blockData.name = line.substring(pos2+1).trim().replace(' ', '_'); blocks.add(blockData); } } finally { in.close(); } } catch (IOException e) { throw new IllegalArgumentException("Can't read block file"); } } }