#region License
/* Copyright (c) 2005 Leslie Sanford
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to
* deal in the Software without restriction, including without limitation the
* rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
* sell copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in
* all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
* THE SOFTWARE.
*/
#endregion
#region Contact
/*
* Leslie Sanford
* Email: jabberdabber@hotmail.com
*/
#endregion
using System;
namespace Sanford.Multimedia.Midi
{
///
/// Provides functionality for building tempo messages.
///
public class TempoChangeBuilder : IMessageBuilder
{
#region TempoChangeBuilder Members
#region Constants
// Value used for shifting bits for packing and unpacking tempo values.
private const int Shift = 8;
#endregion
#region Fields
// The mesage's tempo.
private int tempo = PpqnClock.DefaultTempo;
// The built MetaMessage.
private MetaMessage result = null;
// Indicates whether the tempo property has been changed since
// the last time the message was built.
private bool changed = true;
#endregion
#region Construction
///
/// Initializes a new instance of the TempoChangeBuilder class.
///
public TempoChangeBuilder()
{
}
///
/// Initialize a new instance of the TempoChangeBuilder class with the
/// specified MetaMessage.
///
///
/// The MetaMessage to use for initializing the TempoChangeBuilder class.
///
///
/// If the specified MetaMessage is not a tempo type.
///
///
/// The TempoChangeBuilder uses the specified MetaMessage to initialize
/// its property values.
///
public TempoChangeBuilder(MetaMessage e)
{
Initialize(e);
}
#endregion
#region Methods
///
/// Initializes the TempoChangeBuilder with the specified MetaMessage.
///
///
/// The MetaMessage to use for initializing the TempoChangeBuilder.
///
///
/// If the specified MetaMessage is not a tempo type.
///
public void Initialize(MetaMessage e)
{
#region Require
if(e == null)
{
throw new ArgumentNullException("e");
}
else if(e.MetaType != MetaType.Tempo)
{
throw new ArgumentException("Wrong meta message type.", "e");
}
#endregion
int t = 0;
// If this platform uses little endian byte order.
if(BitConverter.IsLittleEndian)
{
int d = e.Length - 1;
// Pack tempo.
for(int i = 0; i < e.Length; i++)
{
t |= e[d] << (Shift * i);
d--;
}
}
// Else this platform uses big endian byte order.
else
{
// Pack tempo.
for(int i = 0; i < e.Length; i++)
{
t |= e[i] << (Shift * i);
}
}
tempo = t;
}
#endregion
#region Properties
///
/// Gets or sets the tempo.
///
///
/// Value is set to less than zero.
///
public int Tempo
{
get
{
return tempo;
}
set
{
#region Require
if(value < 0)
{
throw new ArgumentOutOfRangeException("Tempo", value,
"Tempo is out of range.");
}
#endregion
tempo = value;
changed = true;
}
}
///
/// Gets the built message.
///
public MetaMessage Result
{
get
{
return result;
}
}
#endregion
#endregion
#region IMessageBuilder Members
///
/// Builds the tempo change MetaMessage.
///
public void Build()
{
// If the tempo has been changed since the last time the message
// was built.
if(changed)
{
byte[] data = new byte[MetaMessage.TempoLength];
// If this platform uses little endian byte order.
if(BitConverter.IsLittleEndian)
{
int d = data.Length - 1;
// Unpack tempo.
for(int i = 0; i < data.Length; i++)
{
data[d] = (byte)(tempo >> (Shift * i));
d--;
}
}
// Else this platform uses big endian byte order.
else
{
// Unpack tempo.
for(int i = 0; i < data.Length; i++)
{
data[i] = (byte)(tempo >> (Shift * i));
}
}
changed = false;
result = new MetaMessage(MetaType.Tempo, data);
}
}
#endregion
}
}