blob: c049ef6c6944f5ebf4aabbc5df62a2f79139a959 [file] [log] [blame]
/*******************************************************************************
* Copyright (c) 2000, 2009 IBM Corporation and others.
* All rights reserved. This program and the accompanying materials
* are made available under the terms of the Eclipse Public License v1.0
* which accompanies this distribution, and is available at
* http://www.eclipse.org/legal/epl-v10.html
*
* Contributors:
* IBM Corporation - initial API and implementation
* Matt McCutchen - add check for EOF handling
******************************************************************************/
package org.eclipse.jdt.internal.formatter.comment;
import java.io.IOException;
import java.io.Reader;
import org.eclipse.jdt.internal.compiler.parser.ScannerHelper;
/**
* Reads the text contents from a reader and computes for each character
* a potential substitution. The substitution may eat more characters than
* only the one passed into the computation routine.
*/
public abstract class SubstitutionTextReader extends Reader {
private Reader fReader;
private boolean fWasWhiteSpace;
private int fCharAfterWhiteSpace;
/**
* Tells whether white space characters are skipped.
*/
private boolean fSkipWhiteSpace= true;
private boolean fReadFromBuffer;
private StringBuffer fBuffer;
private int fIndex;
protected SubstitutionTextReader(Reader reader) {
this.fReader= reader;
this.fBuffer= new StringBuffer();
this.fIndex= 0;
this.fReadFromBuffer= false;
this.fCharAfterWhiteSpace= -1;
this.fWasWhiteSpace= true;
}
/**
* Gets the content as a String
*/
public String getString() throws IOException {
StringBuffer buf= new StringBuffer();
int ch;
while ((ch= read()) != -1) {
buf.append((char)ch);
}
return buf.toString();
}
/**
* Implement to compute the substitution for the given character and
* if necessary subsequent characters. Use <code>nextChar</code>
* to read subsequent characters.
*/
protected abstract String computeSubstitution(int c) throws IOException;
/**
* Returns the internal reader.
*/
protected Reader getReader() {
return this.fReader;
}
/**
* Returns the next character.
*/
protected int nextChar() throws IOException {
this.fReadFromBuffer= (this.fBuffer.length() > 0);
if (this.fReadFromBuffer) {
char ch= this.fBuffer.charAt(this.fIndex++);
if (this.fIndex >= this.fBuffer.length()) {
this.fBuffer.setLength(0);
this.fIndex= 0;
}
return ch;
} else {
int ch= this.fCharAfterWhiteSpace;
if (ch == -1) {
ch= this.fReader.read();
}
if (this.fSkipWhiteSpace && ScannerHelper.isWhitespace((char)ch)) {
do {
ch= this.fReader.read();
} while (ScannerHelper.isWhitespace((char)ch));
if (ch != -1) {
this.fCharAfterWhiteSpace= ch;
return ' ';
}
} else {
this.fCharAfterWhiteSpace= -1;
}
return ch;
}
}
/*
* @see Reader#read()
*/
public int read() throws IOException {
int c;
do {
c= nextChar();
while (!this.fReadFromBuffer && c != -1) {
String s= computeSubstitution(c);
if (s == null)
break;
if (s.length() > 0)
this.fBuffer.insert(0, s);
c= nextChar();
}
} while (this.fSkipWhiteSpace && this.fWasWhiteSpace && (c == ' '));
this.fWasWhiteSpace= (c == ' ' || c == '\r' || c == '\n');
return c;
}
/*
* @see Reader#read(char[],int,int)
*/
public int read(char cbuf[], int off, int len) throws IOException {
int end= off + len;
for (int i= off; i < end; i++) {
int ch= read();
if (ch == -1) {
if (i == off) {
return -1;
} else {
return i - off;
}
}
cbuf[i]= (char)ch;
}
return len;
}
/*
* @see java.io.Reader#ready()
*/
public boolean ready() throws IOException {
return this.fReader.ready();
}
/*
* @see Reader#close()
*/
public void close() throws IOException {
this.fReader.close();
}
/*
* @see Reader#reset()
*/
public void reset() throws IOException {
this.fReader.reset();
this.fWasWhiteSpace= true;
this.fCharAfterWhiteSpace= -1;
this.fBuffer.setLength(0);
this.fIndex= 0;
}
protected final void setSkipWhitespace(boolean state) {
this.fSkipWhiteSpace= state;
}
protected final boolean isSkippingWhitespace() {
return this.fSkipWhiteSpace;
}
}