| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484 | #region Header/** * JsonWriter.cs *   Stream-like facility to output JSON text. * * The authors disclaim copyright to this source code. For more details, see * the COPYING file included with this distribution. **/#endregionusing System;using System.Collections.Generic;using System.Globalization;using System.IO;using System.Text;namespace LitJson{    internal enum Condition    {        InArray,        InObject,        NotAProperty,        Property,        Value    }    internal class WriterContext    {        public int  Count;        public bool InArray;        public bool InObject;        public bool ExpectingValue;        public int  Padding;    }    public class JsonWriter    {        #region Fields        private static readonly NumberFormatInfo number_format;        private WriterContext        context;        private Stack<WriterContext> ctx_stack;        private bool                 has_reached_end;        private char[]               hex_seq;        private int                  indentation;        private int                  indent_value;        private StringBuilder        inst_string_builder;        private bool                 pretty_print;        private bool                 validate;        private bool                 lower_case_properties;        private TextWriter           writer;        #endregion        #region Properties        public int IndentValue {            get { return indent_value; }            set {                indentation = (indentation / indent_value) * value;                indent_value = value;            }        }        public bool PrettyPrint {            get { return pretty_print; }            set { pretty_print = value; }        }        public TextWriter TextWriter {            get { return writer; }        }        public bool Validate {            get { return validate; }            set { validate = value; }        }        public bool LowerCaseProperties {            get { return lower_case_properties; }            set { lower_case_properties = value; }        }        #endregion        #region Constructors        static JsonWriter ()        {            number_format = NumberFormatInfo.InvariantInfo;        }        public JsonWriter ()        {            inst_string_builder = new StringBuilder ();            writer = new StringWriter (inst_string_builder);            Init ();        }        public JsonWriter (StringBuilder sb) :            this (new StringWriter (sb))        {        }        public JsonWriter (TextWriter writer)        {            if (writer == null)                throw new ArgumentNullException ("writer");            this.writer = writer;            Init ();        }        #endregion        #region Private Methods        private void DoValidation (Condition cond)        {            if (! context.ExpectingValue)                context.Count++;            if (! validate)                return;            if (has_reached_end)                throw new JsonException (                    "A complete JSON symbol has already been written");            switch (cond) {            case Condition.InArray:                if (! context.InArray)                    throw new JsonException (                        "Can't close an array here");                break;            case Condition.InObject:                if (! context.InObject || context.ExpectingValue)                    throw new JsonException (                        "Can't close an object here");                break;            case Condition.NotAProperty:                if (context.InObject && ! context.ExpectingValue)                    throw new JsonException (                        "Expected a property");                break;            case Condition.Property:                if (! context.InObject || context.ExpectingValue)                    throw new JsonException (                        "Can't add a property here");                break;            case Condition.Value:                if (! context.InArray &&                    (! context.InObject || ! context.ExpectingValue))                    throw new JsonException (                        "Can't add a value here");                break;            }        }        private void Init ()        {            has_reached_end = false;            hex_seq = new char[4];            indentation = 0;            indent_value = 4;            pretty_print = false;            validate = true;            lower_case_properties = false;            ctx_stack = new Stack<WriterContext> ();            context = new WriterContext ();            ctx_stack.Push (context);        }        private static void IntToHex (int n, char[] hex)        {            int num;            for (int i = 0; i < 4; i++) {                num = n % 16;                if (num < 10)                    hex[3 - i] = (char) ('0' + num);                else                    hex[3 - i] = (char) ('A' + (num - 10));                n >>= 4;            }        }        private void Indent ()        {            if (pretty_print)                indentation += indent_value;        }        private void Put (string str)        {            if (pretty_print && ! context.ExpectingValue)                for (int i = 0; i < indentation; i++)                    writer.Write (' ');            writer.Write (str);        }        private void PutNewline ()        {            PutNewline (true);        }        private void PutNewline (bool add_comma)        {            if (add_comma && ! context.ExpectingValue &&                context.Count > 1)                writer.Write (',');            if (pretty_print && ! context.ExpectingValue)                writer.Write (Environment.NewLine);        }        private void PutString (string str)        {            Put (String.Empty);            writer.Write ('"');            int n = str.Length;            for (int i = 0; i < n; i++) {                switch (str[i]) {                case '\n':                    writer.Write ("\\n");                    continue;                case '\r':                    writer.Write ("\\r");                    continue;                case '\t':                    writer.Write ("\\t");                    continue;                case '"':                case '\\':                    writer.Write ('\\');                    writer.Write (str[i]);                    continue;                case '\f':                    writer.Write ("\\f");                    continue;                case '\b':                    writer.Write ("\\b");                    continue;                }                if ((int) str[i] >= 32 && (int) str[i] <= 126) {                    writer.Write (str[i]);                    continue;                }                // Default, turn into a \uXXXX sequence                IntToHex ((int) str[i], hex_seq);                writer.Write ("\\u");                writer.Write (hex_seq);            }            writer.Write ('"');        }        private void Unindent ()        {            if (pretty_print)                indentation -= indent_value;        }        #endregion        public override string ToString ()        {            if (inst_string_builder == null)                return String.Empty;            return inst_string_builder.ToString ();        }        public void Reset ()        {            has_reached_end = false;            ctx_stack.Clear ();            context = new WriterContext ();            ctx_stack.Push (context);            if (inst_string_builder != null)                inst_string_builder.Remove (0, inst_string_builder.Length);        }        public void Write (bool boolean)        {            DoValidation (Condition.Value);            PutNewline ();            Put (boolean ? "true" : "false");            context.ExpectingValue = false;        }        public void Write (decimal number)        {            DoValidation (Condition.Value);            PutNewline ();            Put (Convert.ToString (number, number_format));            context.ExpectingValue = false;        }        public void Write (double number)        {            DoValidation (Condition.Value);            PutNewline ();            string str = Convert.ToString (number, number_format);            Put (str);            if (str.IndexOf ('.') == -1 &&                str.IndexOf ('E') == -1)                writer.Write (".0");            context.ExpectingValue = false;        }        public void Write(float number)        {            DoValidation(Condition.Value);            PutNewline();            string str = Convert.ToString(number, number_format);            Put(str);            context.ExpectingValue = false;        }        public void Write (int number)        {            DoValidation (Condition.Value);            PutNewline ();            Put (Convert.ToString (number, number_format));            context.ExpectingValue = false;        }        public void Write (long number)        {            DoValidation (Condition.Value);            PutNewline ();            Put (Convert.ToString (number, number_format));            context.ExpectingValue = false;        }        public void Write (string str)        {            DoValidation (Condition.Value);            PutNewline ();            if (str == null)                Put ("null");            else                PutString (str);            context.ExpectingValue = false;        }        [CLSCompliant(false)]        public void Write (ulong number)        {            DoValidation (Condition.Value);            PutNewline ();            Put (Convert.ToString (number, number_format));            context.ExpectingValue = false;        }        public void WriteArrayEnd ()        {            DoValidation (Condition.InArray);            PutNewline (false);            ctx_stack.Pop ();            if (ctx_stack.Count == 1)                has_reached_end = true;            else {                context = ctx_stack.Peek ();                context.ExpectingValue = false;            }            Unindent ();            Put ("]");        }        public void WriteArrayStart ()        {            DoValidation (Condition.NotAProperty);            PutNewline ();            Put ("[");            context = new WriterContext ();            context.InArray = true;            ctx_stack.Push (context);            Indent ();        }        public void WriteObjectEnd ()        {            DoValidation (Condition.InObject);            PutNewline (false);            ctx_stack.Pop ();            if (ctx_stack.Count == 1)                has_reached_end = true;            else {                context = ctx_stack.Peek ();                context.ExpectingValue = false;            }            Unindent ();            Put ("}");        }        public void WriteObjectStart ()        {            DoValidation (Condition.NotAProperty);            PutNewline ();            Put ("{");            context = new WriterContext ();            context.InObject = true;            ctx_stack.Push (context);            Indent ();        }        public void WritePropertyName (string property_name)        {            DoValidation (Condition.Property);            PutNewline ();            string propertyName = (property_name == null || !lower_case_properties)                ? property_name                : property_name.ToLowerInvariant();            PutString (propertyName);            if (pretty_print) {                if (propertyName.Length > context.Padding)                    context.Padding = propertyName.Length;                for (int i = context.Padding - propertyName.Length;                     i >= 0; i--)                    writer.Write (' ');                writer.Write (": ");            } else                writer.Write (':');            context.ExpectingValue = true;        }    }}
 |