1
0
mirror of https://github.com/ppy/osu.git synced 2024-12-21 16:42:55 +08:00
osu-lazer/osu.Game/IO/Legacy/SerializationWriter.cs

250 lines
7.7 KiB
C#
Raw Normal View History

// Copyright (c) ppy Pty Ltd <contact@ppy.sh>. Licensed under the MIT Licence.
// See the LICENCE file in the repository root for full licence text.
2018-04-13 17:19:50 +08:00
2022-06-17 15:37:17 +08:00
#nullable disable
2018-04-13 17:19:50 +08:00
using System;
using System.Collections.Generic;
using System.IO;
using System.Text;
2019-02-28 12:31:40 +08:00
2018-04-13 17:19:50 +08:00
// ReSharper disable ConditionIsAlwaysTrueOrFalse (we're allowing nulls to be passed to the writer where the underlying class doesn't).
// ReSharper disable HeuristicUnreachableCode
namespace osu.Game.IO.Legacy
{
/// <summary> SerializationWriter. Extends BinaryWriter to add additional data types,
/// handle null strings and simplify use with ISerializable. </summary>
public class SerializationWriter : BinaryWriter
{
2021-06-23 14:10:03 +08:00
public SerializationWriter(Stream s, bool leaveOpen = false)
: base(s, Encoding.UTF8, leaveOpen)
2018-04-13 17:19:50 +08:00
{
}
/// <summary> Static method to initialise the writer with a suitable MemoryStream. </summary>
public static SerializationWriter GetWriter()
{
MemoryStream ms = new MemoryStream(1024);
return new SerializationWriter(ms);
}
/// <summary> Writes a string to the buffer. Overrides the base implementation so it can cope with nulls </summary>
public override void Write(string str)
{
if (str == null)
{
Write((byte)ObjType.NullType);
2018-04-13 17:19:50 +08:00
}
else
{
Write((byte)ObjType.StringType);
2018-04-13 17:19:50 +08:00
base.Write(str);
}
}
/// <summary> Writes a byte array to the buffer. Overrides the base implementation to
/// send the length of the array which is needed when it is retrieved </summary>
public override void Write(byte[] b)
{
if (b == null)
{
Write(-1);
}
else
{
int len = b.Length;
Write(len);
if (len > 0) base.Write(b);
}
}
/// <summary> Writes a char array to the buffer. Overrides the base implementation to
/// sends the length of the array which is needed when it is read. </summary>
public override void Write(char[] c)
{
if (c == null)
{
Write(-1);
}
else
{
int len = c.Length;
Write(len);
if (len > 0) base.Write(c);
}
}
/// <summary>
/// Writes DateTime to the buffer.
/// </summary>
/// <param name="dt"></param>
public void Write(DateTime dt)
{
Write(dt.ToUniversalTime().Ticks);
}
/// <summary> Writes a generic ICollection (such as an IList(T)) to the buffer.</summary>
public void Write<T>(List<T> c) where T : ILegacySerializable
{
if (c == null)
{
Write(-1);
}
else
{
int count = c.Count;
Write(count);
for (int i = 0; i < count; i++)
c[i].WriteToStream(this);
}
}
/// <summary> Writes a generic IDictionary to the buffer. </summary>
public void Write<TKey, TValue>(IDictionary<TKey, TValue> d)
2018-04-13 17:19:50 +08:00
{
if (d == null)
{
Write(-1);
}
else
{
Write(d.Count);
2019-04-01 11:16:05 +08:00
foreach (KeyValuePair<TKey, TValue> kvp in d)
2018-04-13 17:19:50 +08:00
{
WriteObject(kvp.Key);
WriteObject(kvp.Value);
}
}
}
/// <summary> Writes an arbitrary object to the buffer. Useful where we have something of type "object"
/// and don't know how to treat it. This works out the best method to use to write to the buffer. </summary>
public void WriteObject(object obj)
{
if (obj == null)
{
Write((byte)ObjType.NullType);
2018-04-13 17:19:50 +08:00
}
else
{
switch (obj)
2018-04-13 17:19:50 +08:00
{
case bool boolObj:
Write((byte)ObjType.BoolType);
Write(boolObj);
2018-04-13 17:19:50 +08:00
break;
case byte byteObj:
Write((byte)ObjType.ByteType);
Write(byteObj);
2018-04-13 17:19:50 +08:00
break;
case ushort ushortObj:
Write((byte)ObjType.UInt16Type);
Write(ushortObj);
2018-04-13 17:19:50 +08:00
break;
case uint uintObj:
Write((byte)ObjType.UInt32Type);
Write(uintObj);
2018-04-13 17:19:50 +08:00
break;
case ulong ulongObj:
Write((byte)ObjType.UInt64Type);
Write(ulongObj);
2018-04-13 17:19:50 +08:00
break;
case sbyte sbyteObj:
Write((byte)ObjType.SByteType);
Write(sbyteObj);
2018-04-13 17:19:50 +08:00
break;
case short shortObj:
Write((byte)ObjType.Int16Type);
Write(shortObj);
2018-04-13 17:19:50 +08:00
break;
case int intObj:
Write((byte)ObjType.Int32Type);
Write(intObj);
2018-04-13 17:19:50 +08:00
break;
case long longObj:
Write((byte)ObjType.Int64Type);
Write(longObj);
2018-04-13 17:19:50 +08:00
break;
case char charObj:
Write((byte)ObjType.CharType);
base.Write(charObj);
2018-04-13 17:19:50 +08:00
break;
case string stringObj:
Write((byte)ObjType.StringType);
base.Write(stringObj);
2018-04-13 17:19:50 +08:00
break;
case float floatObj:
Write((byte)ObjType.SingleType);
Write(floatObj);
2018-04-13 17:19:50 +08:00
break;
case double doubleObj:
Write((byte)ObjType.DoubleType);
Write(doubleObj);
2018-04-13 17:19:50 +08:00
break;
case decimal decimalObj:
Write((byte)ObjType.DecimalType);
Write(decimalObj);
2018-04-13 17:19:50 +08:00
break;
case DateTime dateTimeObj:
Write((byte)ObjType.DateTimeType);
Write(dateTimeObj);
2018-04-13 17:19:50 +08:00
break;
case byte[] byteArray:
Write((byte)ObjType.ByteArrayType);
base.Write(byteArray);
2018-04-13 17:19:50 +08:00
break;
case char[] charArray:
Write((byte)ObjType.CharArrayType);
base.Write(charArray);
2018-04-13 17:19:50 +08:00
break;
default:
2022-04-12 20:28:14 +08:00
throw new IOException("Serialization of arbitrary type is not supported.");
2018-04-13 17:19:50 +08:00
} // switch
} // if obj==null
} // WriteObject
public void WriteRawBytes(byte[] b)
{
base.Write(b);
}
public void WriteByteArray(byte[] b)
{
if (b == null)
{
Write(-1);
}
else
{
int len = b.Length;
Write(len);
if (len > 0) base.Write(b);
}
}
public void WriteUtf8(string str)
{
WriteRawBytes(Encoding.UTF8.GetBytes(str));
}
}
}