2005-02-22 04:36:54 +00:00
/ *
The contents of this file are subject to the THDL Open Community License
Version 1 . 0 ( the " License " ) ; you may not use this file except in compliance
with the License . You may obtain a copy of the License on the THDL web site
( http : //www.thdl.org/).
Software distributed under the License is distributed on an " AS IS " basis ,
WITHOUT WARRANTY OF ANY KIND , either express or implied . See the
License for the specific terms governing rights and limitations under the
License .
The Initial Developer of this software is the Tibetan and Himalayan Digital
2005-07-06 02:19:38 +00:00
Library ( THDL ) . Portions created by the THDL are Copyright 2003 - 2005 THDL .
2005-02-22 04:36:54 +00:00
All Rights Reserved .
Contributor ( s ) : ______________________________________ .
* /
package org.thdl.tib.text.ttt ;
2005-06-20 06:18:00 +00:00
import java.math.BigInteger ;
2005-02-22 04:36:54 +00:00
import java.util.ArrayList ;
/ * *
* This singleton class is able to break up Strings of EWTS text ( for
* example , an entire sutra file ) into tsheg bars , comments , etc .
* Non - Tibetan parts are segregated ( so that consumers can ensure that
* they remain non - Tibetan ) , and Tibetan passages are broken up into
* tsheg bars .
*
* This is not public because you should use { @link EWTSTraits # scanner ( ) } .
*
* @author David Chandler * /
class EWTSTshegBarScanner extends TTshegBarScanner {
2005-06-20 06:18:00 +00:00
/** Returns true iff ch can appear within an EWTS tsheg bar. */
protected static boolean isValidInsideTshegBar ( char ch ) {
// '\\' is absent, but should it be? TODO(DLC)[EWTS->Tibetan]
return ( ( ch > = '0' & & ch < = '9' )
| | ( ch > = '\u0f71' & & ch < = '\u0f84' )
| | EWTSTraits . instance ( ) . isUnicodeConsonant ( ch )
| | EWTSTraits . instance ( ) . isUnicodeWowel ( ch )
| | ( ch > = '\u0f20' & & ch < = '\u0f33' )
| | " khgncjytdpbmtstdzwzz'rlafvTDNSWYReuioIAUMHX?^ \ u0f39 \ u0f35 \ u0f37.+~'`- \ u0f19 \ u0f18 \ u0f3f \ u0f3e \ u0f86 \ u0f87 \ u0f88 " . indexOf ( ch ) > = 0 ) ;
2005-07-06 02:19:38 +00:00
// NOTE: We treat \u0f00 as punctuation, not something valid
// inside a tsheg bar. This is questionable, but since it is
// a tsheg bar all by itself (almost always in practice,
// anyway) and since it would've required code changes I
// didn't want to make, that's how it is.
2005-06-20 06:18:00 +00:00
}
2005-07-06 02:19:38 +00:00
// TODO(dchandler): use jflex, javacc or something similar as much
// as you can. I don't think EWTS can be perfectly parsed by
// javacc, by the way, but having several components in a pipeline
// would likely make things more maintainable.
//
// NOTE: EWTS doesn't fully specify how Unicode escapes (e.g.,
// [\\u0f20] should work). When do you evaluate them?
// Immediately like Java source files or later, say right before
// outputting? Our answer: immediately. [\\u0f88+ka] becomes
// hard to do otherwise. This means we treat actual Unicode in a
// way that a reader of the EWTS standard might not think about,
// but actual Unicode is rare in the input
// (TODO(DLC)[EWTS->Tibetan]: it's so rare that we ought to give a
// warning/error when we see it).
/ * * See the comment in TTshegBarScanner . This does not find
errors and warnings that you ' d think of a parser finding ( TODO ( DLC ) [ EWTS - > Tibetan ] :
DOES IT ? ) . * /
public ArrayList scan ( String s , StringBuffer errors , int maxErrors , // TODO(DLC)[EWTS->Tibetan]: ignored
boolean shortMessages , String warningLevel ) {
// the size depends on whether it's mostly Tibetan or mostly
// Latin and a number of other factors. This is meant to be
// an underestimate, but not too much of an underestimate.
ArrayList al = new ArrayList ( s . length ( ) / 10 ) ;
2005-06-20 06:18:00 +00:00
2005-07-06 02:19:38 +00:00
StringBuffer sb = new StringBuffer ( s ) ;
ExpandEscapeSequences ( sb ) ;
int sl = sb . length ( ) ;
// TODO(DLC)[EWTS->Tibetan]:: '@#', in ewts->tmw, is not working
// TODO(DLC)[EWTS->Tibetan]:: 'jamX 'jam~X one is not working in ->tmw mode
// TODO(DLC)[EWTS->Tibetan]:: dzaHsogs is not working
for ( int i = 0 ; i < sl ; i + + ) {
if ( isValidInsideTshegBar ( sb . charAt ( i ) ) ) {
StringBuffer tbsb = new StringBuffer ( ) ;
for ( ; i < sl ; i + + ) {
if ( isValidInsideTshegBar ( sb . charAt ( i ) ) )
tbsb . append ( sb . charAt ( i ) ) ;
else {
- - i ;
break ;
}
}
al . add ( new TString ( " EWTS " , tbsb . toString ( ) ,
TString . TIBETAN_NON_PUNCTUATION ) ) ;
} else {
// NOTE: It's questionable, but we treat
// \u0f00 like punctuation because it was
// easier coding that way.
if ( ( sb . charAt ( i ) > = EWTSTraits . PUA_MIN
& & sb . charAt ( i ) < = EWTSTraits . PUA_MAX )
| | ( sb . charAt ( i ) > = '\u0f00' & & sb . charAt ( i ) < = '\u0f17' )
| | ( sb . charAt ( i ) > = '\u0f1a' & & sb . charAt ( i ) < = '\u0f1f' )
| | ( sb . charAt ( i ) > = '\u0fbe' & & sb . charAt ( i ) < = '\u0fcc' )
| | ( sb . charAt ( i ) > = '\u0fcf' & & sb . charAt ( i ) < = '\u0fd1' )
| | ( EWTSTraits . SAUVASTIKA = = sb . charAt ( i ) )
| | ( EWTSTraits . SWASTIKA = = sb . charAt ( i ) )
| | ( " /;|!:=_@#$%<>()*& \ r \ n \ t \ u0f36 \ u0f38 \ u0f89 \ u0f8a \ u0f8b " . indexOf ( sb . charAt ( i ) )
> = 0 ) ) {
al . add ( new TString ( " EWTS " , sb . substring ( i , i + 1 ) ,
TString . TIBETAN_PUNCTUATION ) ) ;
} else {
al . add ( new TString ( " EWTS " , " ERROR TODO(DLC)[EWTS->Tibetan]: this character is illegal in EWTS: " + sb . substring ( i , i + 1 ) ,
TString . ERROR ) ) ;
2005-06-20 06:18:00 +00:00
}
2005-07-06 02:19:38 +00:00
}
2005-06-20 06:18:00 +00:00
}
2005-07-06 02:19:38 +00:00
return al ;
}
2005-06-20 06:18:00 +00:00
/ * * Modifies the EWTS in sb such that Unicode escape sequences are
* expanded . * /
public static void ExpandEscapeSequences ( StringBuffer sb ) {
int sl ;
for ( int i = 0 ; i < ( sl = sb . length ( ) ) ; i + + ) {
if ( i + " \\ u00000000 " . length ( ) < = sl ) {
if ( sb . charAt ( i ) = = '\\' & & sb . charAt ( i + 1 ) = = 'u' | | sb . charAt ( i + 1 ) = = 'U' ) {
boolean isEscape = true ;
for ( int j = 0 ; j < " 00000000 " . length ( ) ; j + + ) {
char ch = sb . charAt ( i + " \\ u " . length ( ) + j ) ;
if ( ! ( ( ch < = '9' & & ch > = '0' )
| | ( ch < = 'F' & & ch > = 'A' )
| | ( ch < = 'f' & & ch > = 'a' ) ) ) {
isEscape = false ;
break ;
}
}
if ( isEscape ) {
long x = - 1 ;
try {
BigInteger bigx = new java . math . BigInteger ( sb . substring ( i + 2 , i + 10 ) , 16 ) ;
x = bigx . longValue ( ) ;
if ( ! ( bigx . compareTo ( new BigInteger ( " 0 " , 16 ) ) > = 0
& & bigx . compareTo ( new BigInteger ( " FFFFFFFF " , 16 ) ) < = 0 ) )
x = - 1 ;
} catch ( NumberFormatException e ) {
// leave x == -1
}
if ( x > = 0 & & x < = 0xFFFF ) {
sb . replace ( i , i + " \\ uXXXXyyyy " . length ( ) , new String ( new char [ ] { ( char ) x } ) ) ;
continue ;
} else if ( x > = 0x00000000L
& & x < = 0xFFFFFFFFL ) {
// TODO(DLC)[EWTS->Tibetan]: do nothing? test errors al.add(new TString("EWTS", "Sorry, we don't yet support Unicode escape sequences above 0x0000FFFF! File a bug.",
//TString.ERROR));
i + = " uXXXXYYYY " . length ( ) ;
continue ;
}
}
}
}
if ( i + " \\ u0000 " . length ( ) < = sl ) {
if ( sb . charAt ( i ) = = '\\' & & sb . charAt ( i + 1 ) = = 'u' | | sb . charAt ( i + 1 ) = = 'U' ) {
boolean isEscape = true ;
for ( int j = 0 ; j < " 0000 " . length ( ) ; j + + ) {
char ch = sb . charAt ( i + " \\ u " . length ( ) + j ) ;
if ( ! ( ( ch < = '9' & & ch > = '0' )
| | ( ch < = 'F' & & ch > = 'A' )
| | ( ch < = 'f' & & ch > = 'a' ) ) ) {
isEscape = false ;
break ;
}
}
if ( isEscape ) {
int x = - 1 ;
try {
if ( ! ( ( x = Integer . parseInt ( sb . substring ( i + 2 , i + 6 ) , 16 ) ) > = 0x0000
& & x < = 0xFFFF ) )
x = - 1 ;
} catch ( NumberFormatException e ) {
// leave x == -1
}
if ( x > = 0 ) {
sb . replace ( i , i + " \\ uXXXX " . length ( ) , new String ( new char [ ] { ( char ) x } ) ) ;
continue ;
}
}
}
}
}
2005-02-22 04:36:54 +00:00
}
/** non-public because this is a singleton */
protected EWTSTshegBarScanner ( ) { }
private static EWTSTshegBarScanner singleton = null ;
/** Returns the sole instance of this class. */
public synchronized static EWTSTshegBarScanner instance ( ) {
if ( null = = singleton ) {
singleton = new EWTSTshegBarScanner ( ) ;
}
return singleton ;
}
}