2003-08-10 19:30:07 +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
Library ( THDL ) . Portions created by the THDL are Copyright 2003 THDL .
All Rights Reserved .
Contributor ( s ) : ______________________________________ .
* /
package org.thdl.tib.text.ttt ;
import java.util.ArrayList ;
2003-09-10 01:19:05 +00:00
/ * * A list of non - empty list of { @link TStackListList
2003-08-10 19:30:07 +00:00
* TStackListLists } representing all the ways you could break up a
* tsheg bar of ACIP into stacks ( i . e . , grapheme clusters ) .
*
* @author David Chandler * /
class TParseTree {
/** a fast, non-thread-safe, random-access list implementation: */
private ArrayList al = new ArrayList ( ) ;
/** Creates an empty list. */
public TParseTree ( ) { }
2003-08-12 04:13:11 +00:00
/** Returns the ith list of stack lists in this parse tree. */
2003-08-10 19:30:07 +00:00
public TStackListList get ( int i ) { return ( TStackListList ) al . get ( i ) ; }
/** Adds p to the end of this list. */
public void add ( TStackListList p )
throws IllegalArgumentException
{
if ( p . isEmpty ( ) )
throw new IllegalArgumentException ( " p is empty " ) ;
al . add ( p ) ;
}
/ * * Returns the number of TStackListLists in this list . See
* also { @link # numberOfParses ( ) } , which gives a different
* interpretation of the size of this tree . * /
public int size ( ) { return al . size ( ) ; }
/ * * Returns the number of different parses one could make from
* this parse tree . Returns zero if this list is empty . * /
public int numberOfParses ( ) {
if ( al . isEmpty ( ) ) return 0 ;
int k = 1 ;
int sz = size ( ) ;
for ( int i = 0 ; i < sz ; i + + ) {
k * = get ( i ) . size ( ) ;
}
return k ;
}
2003-09-10 01:19:05 +00:00
/ * * Returns the number of { @link TPair pairs } that are in a
2003-08-10 19:30:07 +00:00
* parse of this tree . * /
public int numberOfPairs ( ) {
if ( al . isEmpty ( ) ) return 0 ;
int k = 1 ;
int sz = size ( ) ;
for ( int i = 0 ; i < sz ; i + + ) {
// get(i).get(0) is the same size as get(i).get(1),
// get(i).get(2), ...
k + = get ( i ) . get ( 0 ) . size ( ) ;
}
return k ;
}
/ * * Returns an iterator that will iterate over the { @link
* # numberOfParses } available . * /
public ParseIterator getParseIterator ( ) {
return new ParseIterator ( al ) ;
}
/ * * Returns a list containing the legal parses of this parse tree .
* By & quot ; legal & quot ; , we mean a sequence of stacks that is
* legal by the rules of Tibetan tsheg bar syntax ( sometimes
* called spelling ) . This will return the { G - YA } parse of { GYA }
* as well as the { GYA } parse , so watch yourself . * /
public TStackListList getLegalParses ( ) {
TStackListList sll = new TStackListList ( 2 ) ; // save memory
ParseIterator pi = getParseIterator ( ) ;
while ( pi . hasNext ( ) ) {
TStackList sl = pi . next ( ) ;
2003-08-23 22:03:37 +00:00
if ( sl . isLegalTshegBar ( false ) . isLegal ) {
2003-08-10 19:30:07 +00:00
sll . add ( sl ) ;
}
}
return sll ;
}
2003-08-12 04:13:11 +00:00
/ * * Returns a list ( never null ) containing the parses of this
* parse tree that are not clearly illegal . * /
2003-08-10 19:30:07 +00:00
public TStackListList getNonIllegalParses ( ) {
TStackListList sll = new TStackListList ( 2 ) ; // save memory
ParseIterator pi = getParseIterator ( ) ;
while ( pi . hasNext ( ) ) {
TStackList sl = pi . next ( ) ;
if ( ! sl . isClearlyIllegal ( ) ) {
sll . add ( sl ) ;
}
}
return sll ;
}
/ * * Returns the best parse , if there is a unique parse that is
* clearly preferred to other parses . Basically , if there ' s a
* unique legal parse , you get it . If there ' s not , but there is
* a unique non - illegal parse , you get it . If there ' s not a
* unique answer , null is returned . * /
public TStackList getBestParse ( ) {
2003-08-23 22:03:37 +00:00
TStackListList up = getUniqueParse ( false ) ;
2003-08-10 19:30:07 +00:00
if ( up . size ( ) = = 1 )
return up . get ( 0 ) ;
2003-08-23 22:03:37 +00:00
2003-08-10 19:30:07 +00:00
up = getNonIllegalParses ( ) ;
int sz = up . size ( ) ;
2003-08-12 04:13:11 +00:00
if ( sz = = 1 ) {
2003-08-10 19:30:07 +00:00
return up . get ( 0 ) ;
2003-08-12 04:13:11 +00:00
} else if ( sz > 1 ) {
2005-06-20 06:18:00 +00:00
// TODO(DLC)[EWTS->Tibetan]: does this still happen? If so, when?
//
// System.out.println("SHO NUFF, >1 non-illegal parses still happens");
2003-08-12 04:13:11 +00:00
// {PADMA}, for example. Our technique is to go from the
// left and stack as much as we can. So {PA}{D}{MA} is
// inferior to {PA}{D+MA}, and {PA}{D+MA}{D}{MA} is
// inferior to {PA}{D+MA}{D+MA}. We do not look for the
// minimum number of glyphs, though -- {PA}{N+D}{B+H+R}
// and {PA}{N}{D+B+H+R} tie by that score, but the former
// is the clear winner.
// We give a warning about these, optionally, so that
// users can produce output that even a dumb ACIP reader
2003-08-24 06:40:53 +00:00
// can understand. See getWarning("All", ..).
2003-08-12 04:13:11 +00:00
// if j is in this list, then up.get(j) is still a
// potential winner.
ArrayList candidates = new ArrayList ( sz ) ;
for ( int i = 0 ; i < sz ; i + + )
candidates . add ( new Integer ( i ) ) ;
boolean keepGoing = true ;
int stackNumber = 0 ;
boolean someoneHasThisStack = true ;
while ( someoneHasThisStack & & candidates . size ( ) > 1 ) {
// maybe none of the candidates have stackNumber+1
// stacks. If none do, we'll quit.
someoneHasThisStack = false ;
int maxGlyphsInThisStack = 0 ;
for ( int k = 0 ; k < candidates . size ( ) ; k + + ) {
TStackList sl = up . get ( ( ( Integer ) candidates . get ( k ) ) . intValue ( ) ) ;
if ( sl . size ( ) > stackNumber ) {
int ng ;
if ( ( ng = sl . get ( stackNumber ) . size ( ) ) > maxGlyphsInThisStack )
maxGlyphsInThisStack = ng ;
someoneHasThisStack = true ;
}
}
// Remove all candidates that aren't keeping up.
if ( someoneHasThisStack ) {
for ( int k = 0 ; k < candidates . size ( ) ; k + + ) {
TStackList sl = up . get ( ( ( Integer ) candidates . get ( k ) ) . intValue ( ) ) ;
if ( sl . size ( ) > stackNumber ) {
if ( sl . get ( stackNumber ) . size ( ) ! = maxGlyphsInThisStack )
candidates . remove ( k - - ) ;
} else throw new Error ( " impossible! " ) ;
}
}
+ + stackNumber ;
}
if ( candidates . size ( ) = = 1 )
return up . get ( ( ( Integer ) candidates . get ( 0 ) ) . intValue ( ) ) ;
else
return null ;
2003-08-10 19:30:07 +00:00
}
return null ;
}
/ * * Returns a list containing the unique legal parse of this parse
2003-10-16 04:15:10 +00:00
* tree if there is a unique legal parse . Returns an empty list
2003-08-10 19:30:07 +00:00
* if there are no legal parses . Returns a list containing all
* legal parses if there two or more equally good parses . By
* & quot ; legal & quot ; , we mean a sequence of stacks that is legal
* by the rules of Tibetan tsheg bar syntax ( sometimes called
2003-08-23 22:03:37 +00:00
* spelling ) .
* @param noPrefixTests true if you want to pretend that every
* stack can take every prefix , which is not the case in
* reality * /
public TStackListList getUniqueParse ( boolean noPrefixTests ) {
2003-09-12 05:06:37 +00:00
// For Sanskrit+Tibetan:
TStackListList allNonillegalParses = new TStackListList ( 2 ) ; // save memory
// For Tibetan only:
TStackListList allStrictlyLegalParses = new TStackListList ( 2 ) ; // save memory
2003-08-10 19:30:07 +00:00
TStackListList legalParsesWithVowelOnRoot = new TStackListList ( 1 ) ;
ParseIterator pi = getParseIterator ( ) ;
while ( pi . hasNext ( ) ) {
TStackList sl = pi . next ( ) ;
2003-09-12 05:06:37 +00:00
BoolTriple bt = sl . isLegalTshegBar ( noPrefixTests ) ;
if ( bt . isLegal ) {
if ( bt . isLegalAndHasAVowelOnRoot )
2003-08-10 19:30:07 +00:00
legalParsesWithVowelOnRoot . add ( sl ) ;
2003-10-26 00:32:55 +00:00
if ( ! bt . isLegalButSanskrit ( ) )
2003-09-12 05:06:37 +00:00
allStrictlyLegalParses . add ( sl ) ;
allNonillegalParses . add ( sl ) ;
2003-08-10 19:30:07 +00:00
}
}
if ( legalParsesWithVowelOnRoot . size ( ) = = 1 )
return legalParsesWithVowelOnRoot ;
else {
2003-09-12 05:06:37 +00:00
if ( allStrictlyLegalParses . size ( ) = = 1 )
return allStrictlyLegalParses ;
if ( allStrictlyLegalParses . size ( ) > 2 )
throw new Error ( " can this happen? " ) ;
2003-08-10 19:30:07 +00:00
if ( legalParsesWithVowelOnRoot . size ( ) = = 2 ) {
2003-10-16 04:15:10 +00:00
if ( legalParsesWithVowelOnRoot . get ( 0 ) . size ( )
! = 1 + legalParsesWithVowelOnRoot . get ( 1 ) . size ( ) ) {
// MARDA is MAR+DA or MA-R-DA -- both are legal if
// noPrefixTests.
return new TStackListList ( ) ;
} else {
// G-YA vs. GYA.
return new TStackListList ( legalParsesWithVowelOnRoot . get ( 1 ) ) ;
}
2003-08-10 19:30:07 +00:00
}
2003-09-12 05:06:37 +00:00
if ( allNonillegalParses . size ( ) = = 2 ) {
2003-10-16 04:15:10 +00:00
if ( allNonillegalParses . get ( 0 ) . size ( ) ! = 1 + allNonillegalParses . get ( 1 ) . size ( ) ) {
// BDREN, e.g., if noPrefixTests:
return new TStackListList ( ) ;
}
2003-09-12 05:06:37 +00:00
return new TStackListList ( allNonillegalParses . get ( 1 ) ) ;
2003-08-10 19:30:07 +00:00
}
2003-09-12 05:06:37 +00:00
return allNonillegalParses ;
2003-08-10 19:30:07 +00:00
}
}
/** Returns a human-readable representation. */
public String toString ( ) {
return al . toString ( ) ;
}
2005-02-21 01:16:10 +00:00
/ * * Returns true if and only if either x is a TParseTree
2003-08-10 19:30:07 +00:00
* object representing the same TPairLists in the same order
* or x is a String that is equals to the result of { @link
* # toString ( ) } . * /
public boolean equals ( Object x ) {
if ( x instanceof TParseTree ) {
return al . equals ( ( ( TParseTree ) x ) . al ) ;
} else if ( x instanceof String ) {
return toString ( ) . equals ( x ) ;
}
return false ;
}
2003-08-12 04:13:11 +00:00
/ * * Returns null if this parse tree is perfectly legal and valid .
* Returns a warning for users otherwise . If and only if
2003-08-24 06:40:53 +00:00
* warningLevel is " All " , then even unambiguous ACIP like PADMA ,
* which could be improved by being written as PAD + MA , will cause
* a warning .
* @param warningLevel " All " if you ' re paranoid , " Most " to see
* warnings about lacking vowels on final stacks , " Some " to see
* warnings about lacking vowels on non - final stacks and also
* warnings about when prefix rules affect you , " None " if you
2004-04-24 17:49:16 +00:00
* like to see IllegalArgumentExceptions thrown . ( Actually , this
* refers only to the default values - - the level at which any
* particular warning appears is customizable . )
2003-08-12 04:13:11 +00:00
* @param pl the pair list from which this parse tree originated
* @param originalACIP the original ACIP , or null if you want
2004-04-24 17:49:16 +00:00
* this parse tree to make a best guess .
* @param shortMessages true iff you want short error and warning
* messages * /
2003-08-24 06:40:53 +00:00
public String getWarning ( String warningLevel ,
2003-08-12 04:13:11 +00:00
TPairList pl ,
2004-04-24 17:49:16 +00:00
String originalACIP ,
2005-06-20 06:18:00 +00:00
boolean shortMessages ,
TTraits traits ) {
2004-04-24 17:49:16 +00:00
// ROOM_FOR_IMPROVEMENT: Allow one tsheg bar to have multiple
// warnings/errors associated with it. Make this a private
// subroutine, and have the public getWarning(..) call on this
// subroutine again and again until no new error is found. If
// call N yields warning 506, then disable 506 and call again.
// If you get 508, call again, etc. Finally, restore 506
// etc. and return the concatenation of messages 506 and 508.
// {DGYAM--S} should yield both 505 and 509.
if ( ! ErrorsAndWarnings . warningLevelIsKnown ( warningLevel ) )
2003-08-24 06:40:53 +00:00
throw new IllegalArgumentException ( " warning level bad: is it interned? " ) ;
2003-08-23 22:03:37 +00:00
2003-10-21 00:19:15 +00:00
TStackList bestParse = getBestParse ( ) ;
2003-08-23 22:03:37 +00:00
{
TStackListList noPrefixTestsUniqueParse = getUniqueParse ( true ) ;
if ( noPrefixTestsUniqueParse . size ( ) = = 1
& & ! noPrefixTestsUniqueParse . get ( 0 ) . equals ( bestParse ) ) {
2004-04-24 17:49:16 +00:00
if ( ErrorsAndWarnings . isEnabled ( 501 , warningLevel ) )
if ( shortMessages )
return " 501: Using " + bestParse + " , not " + noPrefixTestsUniqueParse . get ( 0 ) ;
else
2005-06-20 06:18:00 +00:00
return " 501: Using " + bestParse + ( ( null ! = originalACIP ) ? ( " for the " + traits . shortTranslitName ( ) + " { " + originalACIP + " } " ) : " " ) + " , but only because the tool's knowledge of prefix rules (see the documentation) says that " + noPrefixTestsUniqueParse . get ( 0 ) + " is not a legal Tibetan tsheg bar ( \" syllable \" ) " ;
2003-08-23 22:03:37 +00:00
}
}
2004-04-24 17:49:16 +00:00
String translit = ( null ! = originalACIP ) ? originalACIP : recoverACIP ( ) ;
2003-08-23 22:03:37 +00:00
TStackListList up = getUniqueParse ( false ) ;
2003-08-12 04:13:11 +00:00
if ( null = = up | | up . size ( ) ! = 1 ) {
boolean isLastStack [ ] = new boolean [ 1 ] ;
TStackListList nip = getNonIllegalParses ( ) ;
if ( nip . size ( ) ! = 1 ) {
2003-10-21 00:19:15 +00:00
if ( null = = bestParse ) {
2004-04-24 17:49:16 +00:00
/ * FIXME : Is this case possible ? We can get to it
in unit testing ( and we do ) , but is there any
ACIP input file that will cause this ? * /
// FIXME: IS 101 NOT TREATED AS AN error, BUT
// INSTEAD TREATED AS A warning?
//
// FIXME: The caller will prepend "WARNING " to this error!
if ( ErrorsAndWarnings . isEnabled ( 101 , warningLevel ) )
return ErrorsAndWarnings . getMessage ( 101 , shortMessages ,
2005-06-20 06:18:00 +00:00
translit ,
traits ) ;
2003-08-12 04:13:11 +00:00
} else {
2005-06-20 09:30:35 +00:00
if ( bestParse . hasStackWithoutVowel ( traits . isACIP ( ) ,
pl , isLastStack ) ) {
2003-08-12 04:13:11 +00:00
if ( isLastStack [ 0 ] ) {
2004-04-24 17:49:16 +00:00
if ( ErrorsAndWarnings . isEnabled ( 502 , warningLevel ) )
return ErrorsAndWarnings . getMessage ( 502 , shortMessages ,
2005-06-20 06:18:00 +00:00
translit ,
traits ) ;
2003-08-12 04:13:11 +00:00
} else {
2005-06-20 09:30:35 +00:00
if ( traits . isACIP ( ) )
throw new Error ( " Can't happen now that we stack greedily " ) ;
2003-08-12 04:13:11 +00:00
}
}
2004-04-24 17:49:16 +00:00
if ( ErrorsAndWarnings . isEnabled ( 503 , warningLevel ) )
return ErrorsAndWarnings . getMessage ( 503 , shortMessages ,
2005-06-20 06:18:00 +00:00
translit ,
traits ) ;
2003-08-12 04:13:11 +00:00
}
} else {
2005-06-20 09:30:35 +00:00
if ( nip . get ( 0 ) . hasStackWithoutVowel ( traits . isACIP ( ) ,
pl , isLastStack ) ) {
2003-08-12 04:13:11 +00:00
if ( isLastStack [ 0 ] ) {
2004-04-24 17:49:16 +00:00
if ( ErrorsAndWarnings . isEnabled ( 502 , warningLevel ) )
return ErrorsAndWarnings . getMessage ( 502 , shortMessages ,
2005-06-20 06:18:00 +00:00
translit ,
traits ) ;
2003-08-12 04:13:11 +00:00
} else {
2005-06-20 09:30:35 +00:00
if ( traits . isACIP ( ) )
throw new Error ( " Can't happen now that we stack greedily [2] " ) ;
2003-08-12 04:13:11 +00:00
}
}
}
}
2003-10-21 02:00:49 +00:00
// Check for things like DZHDZ+H: stacks that have some pluses
// but not all pluses.
//
// Check for things like TSNYA: stacks that could be
// mistransliterations of T+S+N+YA
//
// Check for useless disambiguators.
{
int plnum = 0 ;
while ( plnum < pl . size ( ) & & pl . get ( plnum ) . isDisambiguator ( ) ) {
+ + plnum ;
2004-04-24 17:49:16 +00:00
if ( ErrorsAndWarnings . isEnabled ( 505 , warningLevel ) )
return ErrorsAndWarnings . getMessage ( 505 , shortMessages ,
2005-06-20 06:18:00 +00:00
translit ,
traits ) ;
2003-10-21 02:00:49 +00:00
}
2004-04-24 17:49:16 +00:00
plnum = 0 ;
2003-10-21 02:00:49 +00:00
for ( int stackNum = 0 ; stackNum < bestParse . size ( ) ; stackNum + + ) {
TPairList stack = bestParse . get ( stackNum ) ;
int type = 0 ;
int stackSize = stack . size ( ) ;
boolean hasAmbiguousConsonant = false ; // TS could be TSA or T+SA, so it's "ambiguous"
2004-05-01 20:49:53 +00:00
boolean highPriority507 = false ;
2003-10-21 02:00:49 +00:00
for ( int j = 0 ; j < stackSize ; j + + ) {
TPair tp = pl . get ( plnum + + ) ;
if ( j + 1 < stack . size ( ) ) {
if ( null = = tp . getRight ( ) ) {
if ( type = = 0 )
type = - 1 ;
else if ( type = = 1 )
2004-04-24 17:49:16 +00:00
if ( ErrorsAndWarnings . isEnabled ( 506 , warningLevel ) )
return ErrorsAndWarnings . getMessage ( 506 , shortMessages ,
2005-06-20 06:18:00 +00:00
translit ,
traits ) ;
2003-10-21 02:00:49 +00:00
} else {
if ( type = = 0 )
type = 1 ;
else if ( type = = - 1 )
2004-04-24 17:49:16 +00:00
if ( ErrorsAndWarnings . isEnabled ( 506 , warningLevel ) )
return ErrorsAndWarnings . getMessage ( 506 , shortMessages ,
2005-06-20 06:18:00 +00:00
translit ,
traits ) ;
2003-10-21 02:00:49 +00:00
}
}
if ( stackSize > 1 & & tp . getLeft ( ) ! = null & & tp . getLeft ( ) . length ( ) > 1 ) {
2004-05-01 20:49:53 +00:00
if ( null ! = originalACIP
& & ( originalACIP . startsWith ( " NNY " )
| | originalACIP . startsWith ( " NGH " )
| | originalACIP . startsWith ( " GHNY " )
| | originalACIP . startsWith ( " TNY " )
| | originalACIP . startsWith ( " TSN " ) // and TSNY
| | originalACIP . startsWith ( " HNY " )
| | originalACIP . startsWith ( " TSM " ) // and TSMY
| | originalACIP . startsWith ( " TSY " )
| | originalACIP . startsWith ( " TSR " )
| | originalACIP . startsWith ( " NTS " )
| | originalACIP . startsWith ( " TSTH " )
| | originalACIP . startsWith ( " TSV " )
| | originalACIP . startsWith ( " RTS " ) // and RTSN and RTSNY
// || originalACIP.startsWith("GNY") ... no, GNYA is seen as G-NYA, not G+NYA. FIXME 946058: give warning 512 for {K-GNY}, {BAGNYE}, etc.
// || originalACIP.startsWith("MNY") ... and likewise for MNY.
) ) {
/ *
TM and TMW have glyphs for these :
t + s + th
t + s + r
t + s + w ( i . e . , ACIP { T + S + V } )
r + t + s
r + t + s + n
r + t + s + n + y
n + n + y
n + g + h
g + n + y
g + h + n + y
t + n + y
t + s + n + y
ts + ny
ts + n + y
h + n + y
m + n + y
t + s + m
t + s + y
t + s + r
n + t + s
* /
highPriority507 = true ;
}
2004-04-14 05:44:51 +00:00
// DLC FIXME: gives a false positive warning for Rsh
2003-10-21 02:00:49 +00:00
hasAmbiguousConsonant = true ;
}
}
if ( hasAmbiguousConsonant & & - 1 = = type ) {
2004-05-01 20:49:53 +00:00
int warningNum = ( highPriority507 ) ? 512 : 507 ;
if ( ErrorsAndWarnings . isEnabled ( warningNum , warningLevel ) )
return ErrorsAndWarnings . getMessage ( warningNum ,
shortMessages ,
2005-06-20 06:18:00 +00:00
translit ,
traits ) ;
2003-10-21 02:00:49 +00:00
}
2004-05-01 20:49:53 +00:00
2003-10-21 02:00:49 +00:00
while ( plnum < pl . size ( ) & & pl . get ( plnum ) . isDisambiguator ( ) ) {
+ + plnum ;
2004-04-24 17:49:16 +00:00
if ( ErrorsAndWarnings . isEnabled ( 505 , warningLevel ) )
return ErrorsAndWarnings . getMessage ( 505 , shortMessages ,
2005-06-20 06:18:00 +00:00
translit ,
traits ) ;
2003-10-21 02:00:49 +00:00
}
}
}
2003-10-21 03:36:57 +00:00
// Check for DBA, DBE, DBIm:, etc. (i.e., DB*), BD*, DG*, DM* GD*, DN*, MN*, DGR*, DGY*
if ( pl . size ( ) > = 3 ) {
String left , middle , right ;
left = pl . get ( 0 ) . getLeft ( ) ;
middle = pl . get ( 1 ) . getLeft ( ) ;
right = pl . get ( 2 ) . getLeft ( ) ;
if ( pl . get ( 0 ) . getRight ( ) = = null
& & ! pl . get ( 1 ) . endsACIPStack ( )
& & pl . get ( 2 ) . endsACIPStack ( )
& & null ! = left & & null ! = right ) {
if ( ( " D " . equals ( left ) & & " G " . equals ( middle ) & & " R " . equals ( right ) )
| | ( " D " . equals ( left ) & & " G " . equals ( middle ) & & " Y " . equals ( right ) ) ) {
2004-04-24 17:49:16 +00:00
if ( pl . size ( ) = = 3 ) {
if ( ErrorsAndWarnings . isEnabled ( 508 , warningLevel ) )
return ErrorsAndWarnings . getMessage ( 508 , shortMessages ,
2005-06-20 06:18:00 +00:00
translit ,
traits ) ;
2004-04-24 17:49:16 +00:00
} else {
if ( ErrorsAndWarnings . isEnabled ( 509 , warningLevel ) )
return ErrorsAndWarnings . getMessage ( 509 , shortMessages ,
2005-06-20 06:18:00 +00:00
translit ,
traits ) ;
2004-04-24 17:49:16 +00:00
}
2003-10-21 03:36:57 +00:00
}
}
}
if ( pl . size ( ) > = 2 ) {
String left , right ;
left = pl . get ( 0 ) . getLeft ( ) ;
right = pl . get ( 1 ) . getLeft ( ) ;
if ( pl . get ( 0 ) . getRight ( ) = = null & & pl . get ( 1 ) . endsACIPStack ( )
& & null ! = left & & null ! = right ) {
if ( ( " D " . equals ( left ) & & " B " . equals ( right ) )
| | ( " B " . equals ( left ) & & " D " . equals ( right ) )
| | ( " D " . equals ( left ) & & " G " . equals ( right ) )
| | ( " D " . equals ( left ) & & " M " . equals ( right ) )
| | ( " G " . equals ( left ) & & " D " . equals ( right ) )
| | ( " D " . equals ( left ) & & " N " . equals ( right ) )
| | ( " M " . equals ( left ) & & " N " . equals ( right ) ) ) {
2004-04-24 17:49:16 +00:00
if ( pl . size ( ) = = 2 ) {
if ( ErrorsAndWarnings . isEnabled ( 508 , warningLevel ) )
return ErrorsAndWarnings . getMessage ( 508 , shortMessages ,
2005-06-20 06:18:00 +00:00
translit ,
traits ) ;
2004-04-24 17:49:16 +00:00
} else {
if ( ErrorsAndWarnings . isEnabled ( 509 , warningLevel ) )
return ErrorsAndWarnings . getMessage ( 509 , shortMessages ,
2005-06-20 06:18:00 +00:00
translit ,
traits ) ;
2004-04-24 17:49:16 +00:00
}
2003-10-21 03:36:57 +00:00
}
}
}
2003-08-12 04:13:11 +00:00
return null ;
}
/ * * Returns something akin to the ACIP input ( okay , maybe 1 - 2 - 3 - 4
* instead of 1234 , and maybe AUTPA instead of AUT - PA )
* corresponding to this parse tree . * /
2005-06-20 06:18:00 +00:00
public String recoverACIP ( ) { // TODO(DLC)[EWTS->Tibetan]: acip-specific
2003-08-12 04:13:11 +00:00
ParseIterator pi = getParseIterator ( ) ;
if ( pi . hasNext ( ) ) {
return pi . next ( ) . recoverACIP ( ) ;
}
return null ;
}
2003-08-10 19:30:07 +00:00
/ * * Returns a hashCode appropriate for use with our { @link
* # equals ( Object ) } method . * /
public int hashCode ( ) { return al . hashCode ( ) ; }
2003-08-12 04:13:11 +00:00
/** Returns true if and only if this parse tree is empty. */
public boolean isEmpty ( ) { return al . isEmpty ( ) ; }
2003-08-10 19:30:07 +00:00
}