/*******************************************************************************
* Copyright (c) 2013 EclipseSource.
* 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:
* Ralf Sternberg - initial implementation and API
******************************************************************************/
package com.eclipsesource.json;
import java.io.IOException;
import java.io.Writer;
class JsonWriter {
private static final int CONTROL_CHARACTERS_START = 0x0000;
private static final int CONTROL_CHARACTERS_END = 0x001f;
private static final char[] QUOT_CHARS = { '\\', '"' };
private static final char[] BS_CHARS = { '\\', '\\' };
private static final char[] LF_CHARS = { '\\', 'n' };
private static final char[] CR_CHARS = { '\\', 'r' };
private static final char[] TAB_CHARS = { '\\', 't' };
// In JavaScript, U+2028 and U+2029 characters count as line endings and must be encoded.
// http://stackoverflow.com/questions/2965293/javascript-parse-error-on-u2028-unicode-character
private static final char[] UNICODE_2028_CHARS = { '\\', 'u', '2', '0', '2', '8' };
private static final char[] UNICODE_2029_CHARS = { '\\', 'u', '2', '0', '2', '9' };
private static final char[] HEX_DIGITS = { '0', '1', '2', '3', '4', '5', '6', '7', '8', '9',
'a', 'b', 'c', 'd', 'e', 'f' };
protected final Writer writer;
JsonWriter( Writer writer ) {
this.writer = writer;
}
void write( String string ) throws IOException {
writer.write( string );
}
void writeString( String string ) throws IOException {
writer.write( '"' );
int length = string.length();
int start = 0;
char[] chars = new char[ length ];
string.getChars( 0, length, chars, 0 );
for( int index = 0; index < length; index++ ) {
char[] replacement = getReplacementChars( chars[index] );
if( replacement != null ) {
writer.write( chars, start, index - start );
writer.write( replacement );
start = index+1;
}
}
writer.write( chars, start, length - start );
writer.write( '"' );
}
private static char[] getReplacementChars( char ch ) {
char[] replacement = null;
if( ch == '"' ) {
replacement = QUOT_CHARS;
} else if( ch == '\\' ) {
replacement = BS_CHARS;
} else if( ch == '\n' ) {
replacement = LF_CHARS;
} else if( ch == '\r' ) {
replacement = CR_CHARS;
} else if( ch == '\t' ) {
replacement = TAB_CHARS;
} else if( ch == '\u2028' ) {
replacement = UNICODE_2028_CHARS;
} else if( ch == '\u2029' ) {
replacement = UNICODE_2029_CHARS;
} else if( ch >= CONTROL_CHARACTERS_START && ch <= CONTROL_CHARACTERS_END ) {
replacement = new char[] { '\\', 'u', '0', '0', '0', '0' };
replacement[4] = HEX_DIGITS[ ch >> 4 & 0x000f ];
replacement[5] = HEX_DIGITS[ ch & 0x000f ];
}
return replacement;
}
protected void writeObject( JsonObject object ) throws IOException {
writeBeginObject();
boolean first = true;
for( JsonObject.Member member : object ) {
if( !first ) {
writeObjectValueSeparator();
}
writeString( member.getName() );
writeNameValueSeparator();
member.getValue().write( this );
first = false;
}
writeEndObject();
}
protected void writeBeginObject() throws IOException {
writer.write( '{' );
}
protected void writeEndObject() throws IOException {
writer.write( '}' );
}
protected void writeNameValueSeparator() throws IOException {
writer.write( ':' );
}
protected void writeObjectValueSeparator() throws IOException {
writer.write( ',' );
}
protected void writeArray( JsonArray array ) throws IOException {
writeBeginArray();
boolean first = true;
for( JsonValue value : array ) {
if( !first ) {
writeArrayValueSeparator();
}
value.write( this );
first = false;
}
writeEndArray();
}
protected void writeBeginArray() throws IOException {
writer.write( '[' );
}
protected void writeEndArray() throws IOException {
writer.write( ']' );
}
protected void writeArrayValueSeparator() throws IOException {
writer.write( ',' );
}
}