blob: 519ad9e93b3c7922913cb743337a9ed179b60fee [file] [log] [blame]
/*
* Copyright (c) 2003, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation. Oracle designates this
* particular file as subject to the "Classpath" exception as provided
* by Oracle in the LICENSE file that accompanied this code.
*
* This code is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
* or visit www.oracle.com if you need additional information or have any
* questions.
*/
/*
* (C) Copyright IBM Corp. 2003 - All Rights Reserved
*
* The original version of this source code and documentation is
* copyrighted and owned by IBM. These materials are provided
* under terms of a License Agreement between IBM and Sun.
* This technology is protected by multiple US and International
* patents. This notice and attribution to IBM may not be removed.
*/
package sun.text;
import java.text.CharacterIterator;
public abstract class CodePointIterator {
public static final int DONE = -1;
public abstract void setToStart();
public abstract void setToLimit();
public abstract int next();
public abstract int prev();
public abstract int charIndex();
public static CodePointIterator create(char[] text) {
return new CharArrayCodePointIterator(text);
}
public static CodePointIterator create(char[] text, int start, int limit) {
return new CharArrayCodePointIterator(text, start, limit);
}
public static CodePointIterator create(CharSequence text) {
return new CharSequenceCodePointIterator(text);
}
public static CodePointIterator create(CharacterIterator iter) {
return new CharacterIteratorCodePointIterator(iter);
}
}
final class CharArrayCodePointIterator extends CodePointIterator {
private char[] text;
private int start;
private int limit;
private int index;
public CharArrayCodePointIterator(char[] text) {
this.text = text;
this.limit = text.length;
}
public CharArrayCodePointIterator(char[] text, int start, int limit) {
if (start < 0 || limit < start || limit > text.length) {
throw new IllegalArgumentException();
}
this.text = text;
this.start = this.index = start;
this.limit = limit;
}
public void setToStart() {
index = start;
}
public void setToLimit() {
index = limit;
}
public int next() {
if (index < limit) {
char cp1 = text[index++];
if (Character.isHighSurrogate(cp1) && index < limit) {
char cp2 = text[index];
if (Character.isLowSurrogate(cp2)) {
++index;
return Character.toCodePoint(cp1, cp2);
}
}
return cp1;
}
return DONE;
}
public int prev() {
if (index > start) {
char cp2 = text[--index];
if (Character.isLowSurrogate(cp2) && index > start) {
char cp1 = text[index - 1];
if (Character.isHighSurrogate(cp1)) {
--index;
return Character.toCodePoint(cp1, cp2);
}
}
return cp2;
}
return DONE;
}
public int charIndex() {
return index;
}
}
final class CharSequenceCodePointIterator extends CodePointIterator {
private CharSequence text;
private int index;
public CharSequenceCodePointIterator(CharSequence text) {
this.text = text;
}
public void setToStart() {
index = 0;
}
public void setToLimit() {
index = text.length();
}
public int next() {
if (index < text.length()) {
char cp1 = text.charAt(index++);
if (Character.isHighSurrogate(cp1) && index < text.length()) {
char cp2 = text.charAt(index+1);
if (Character.isLowSurrogate(cp2)) {
++index;
return Character.toCodePoint(cp1, cp2);
}
}
return cp1;
}
return DONE;
}
public int prev() {
if (index > 0) {
char cp2 = text.charAt(--index);
if (Character.isLowSurrogate(cp2) && index > 0) {
char cp1 = text.charAt(index - 1);
if (Character.isHighSurrogate(cp1)) {
--index;
return Character.toCodePoint(cp1, cp2);
}
}
return cp2;
}
return DONE;
}
public int charIndex() {
return index;
}
}
// note this has different iteration semantics than CharacterIterator
final class CharacterIteratorCodePointIterator extends CodePointIterator {
private CharacterIterator iter;
public CharacterIteratorCodePointIterator(CharacterIterator iter) {
this.iter = iter;
}
public void setToStart() {
iter.setIndex(iter.getBeginIndex());
}
public void setToLimit() {
iter.setIndex(iter.getEndIndex());
}
public int next() {
char cp1 = iter.current();
if (cp1 != CharacterIterator.DONE) {
char cp2 = iter.next();
if (Character.isHighSurrogate(cp1) && cp2 != CharacterIterator.DONE) {
if (Character.isLowSurrogate(cp2)) {
iter.next();
return Character.toCodePoint(cp1, cp2);
}
}
return cp1;
}
return DONE;
}
public int prev() {
char cp2 = iter.previous();
if (cp2 != CharacterIterator.DONE) {
if (Character.isLowSurrogate(cp2)) {
char cp1 = iter.previous();
if (Character.isHighSurrogate(cp1)) {
return Character.toCodePoint(cp1, cp2);
}
iter.next();
}
return cp2;
}
return DONE;
}
public int charIndex() {
return iter.getIndex();
}
}