#region --- License ---
/* Licensed under the MIT/X11 license.
* Copyright (c) 2006-2008 the OpenTK Team.
* This notice may not be removed from any source distribution.
* See license.txt for licensing details.
*/
#endregion
using System;
using System.Collections.Generic;
using System.Text;
using System.IO;
namespace OpenTK.Audio
{
///
/// Encapsulates a sound stream and provides decoding and streaming capabilities.
///
public class AudioReader : IDisposable
{
static object reader_lock = new object();
static List readers = new List();
bool disposed;
Stream stream;
AudioReader implementation;
#region --- Constructors ---
#region static AudioReader()
static AudioReader()
{
// TODO: Plugin architecture for sound formats. This is overkill now that we only have a WaveReader (future proofing).
readers.Add(new WaveReader());
}
#endregion
#region protected AudioReader()
protected AudioReader() { }
#endregion
#region public AudioReader(string filename)
/// Creates a new AudioReader that can read the specified sound file.
/// The path to the sound file.
/// A new OpenTK.Audio.AudioReader, which can be used to read from the specified sound file.
public AudioReader(string filename)
: this(new FileStream(filename, FileMode.Open, FileAccess.Read, FileShare.ReadWrite))
{ }
#endregion
#region public AudioReader(Stream s)
/// Creates a new AudioReader that can read the specified soundstream.
/// The System.IO.Stream to read from.
/// A new OpenTK.Audio.AudioReader, which can be used to read from the specified sound stream.
public AudioReader(Stream s)
{
try
{
lock (reader_lock)
{
foreach (AudioReader reader in readers)
{
long pos = s.Position;
if (reader.Supports(s))
{
s.Position = pos;
implementation = (AudioReader)
reader.GetType().GetConstructor(
System.Reflection.BindingFlags.NonPublic | System.Reflection.BindingFlags.Public |
System.Reflection.BindingFlags.Instance,
null,
new Type[] { typeof(Stream) },
null)
.Invoke(new object[] { s });
return;
}
s.Position = pos;
}
}
}
catch (Exception)
{
s.Close();
throw;
}
throw new NotSupportedException("Could not find a decoder for the specified sound stream.");
}
#endregion
#endregion
#region --- Public Members ---
#region public virtual bool Supports(Stream s)
/// When overriden in a derived class, checks if the decoder supports the specified sound stream.
/// The System.IO.Stream to check.
/// True if the sound stream is supported; false otherwise.
public virtual bool Supports(Stream s)
{
if (implementation != null)
return implementation.Supports(s);
throw new NotImplementedException();
}
#endregion
#region public virtual SoundData ReadSamples(long count)
///
/// When overriden in a derived class, reads and decodes the specified number of samples from the sound stream.
///
/// The number of samples to read and decode.
/// An OpenTK.Audio.SoundData object that contains the decoded buffer.
public virtual SoundData ReadSamples(long count)
{
if (implementation != null)
return implementation.ReadSamples(count);
throw new NotImplementedException();
}
#endregion
#region public virtual SoundData ReadToEnd()
///
/// When overriden in a derived class, reads and decodes the sound stream.
///
/// An OpenTK.Audio.SoundData object that contains the decoded buffer.
public virtual SoundData ReadToEnd()
{
if (implementation != null)
return implementation.ReadToEnd();
throw new NotImplementedException();
}
#endregion
#region public virtual int Frequency
///
///
///
public virtual int Frequency
{
get
{
if (implementation != null)
return implementation.Frequency;
else
throw new NotImplementedException();
}
protected set
{
if (implementation != null)
implementation.Frequency = value;
else
throw new NotImplementedException();
}
}
#endregion
#region public virtual bool EndOfFile
public virtual bool EndOfFile
{
get
{
if (implementation != null)
return implementation.EndOfFile;
return this.Stream.Position >= this.Stream.Length;
}
}
#endregion
#endregion
#region --- Protected Members ---
protected virtual Stream Stream
{
get { return stream; }
set { stream = value; }
}
#endregion
#region IDisposable Members
/// Closes the underlying Stream and disposes of the AudioReader resources.
public virtual void Dispose()
{
this.Dispose(true);
GC.SuppressFinalize(this);
}
void Dispose(bool manual)
{
if (!disposed)
{
if (manual)
if (this.Stream != null)
this.Stream.Close();
disposed = true;
}
}
~AudioReader()
{
this.Dispose(false);
}
#endregion
}
}