//
// Properties.cs: This class implements IAudioCodec and IVideoCodec
// and combines codecs to create generic media properties for a file.
//
// Author:
// Brian Nickel (brian.nickel@gmail.com)
//
// Original Source:
// audioproperties.cpp from TagLib
//
// Copyright (C) 2006,2007 Brian Nickel
// Copyright (C) 2003 Scott Wheeler (Original Implementation)
//
// This library is free software; you can redistribute it and/or modify
// it under the terms of the GNU Lesser General Public License version
// 2.1 as published by the Free Software Foundation.
//
// This library is distributed in the hope that it will be useful, but
// WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
// Lesser General Public License for more details.
//
// You should have received a copy of the GNU Lesser General Public
// License along with this library; if not, write to the Free Software
// Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
// USA
//
using System;
using System.Collections.Generic;
using System.Text;
namespace TagLib
{
///
/// This class implements , and
/// and combines codecs to create generic media properties
/// for a file.
///
public class Properties : IAudioCodec, IVideoCodec, IPhotoCodec
{
#region Private Fields
///
/// Contains the codecs.
///
readonly ICodec[] codecs = new ICodec[0];
///
/// Contains the duration.
///
TimeSpan duration = TimeSpan.Zero;
#endregion
#region Constructors
///
/// Constructs and initializes a new instance of with no codecs or duration.
///
///
/// This constructor is used when media properties are
/// not read.
///
public Properties ()
{
}
///
/// Constructs and initializes a new instance of with a specified duration and array
/// of codecs.
///
///
/// A containing the duration of the
/// media, or if the duration is
/// to be read from the codecs.
///
///
/// A containing the codecs to be
/// used in the new instance.
///
public Properties (TimeSpan duration, params ICodec[] codecs)
{
this.duration = duration;
if (codecs != null)
this.codecs = codecs;
}
///
/// Constructs and initializes a new instance of with a specified duration and
/// enumaration of codecs.
///
///
/// A containing the duration of the
/// media, or if the duration is
/// to be read from the codecs.
///
///
/// A object containing the
/// codec to be used in the new instance.
///
public Properties (TimeSpan duration, IEnumerable codecs)
{
this.duration = duration;
if (codecs != null)
this.codecs = new List (codecs).ToArray ();
}
#endregion
#region Public Properties
///
/// Gets the codecs contained in the current instance.
///
///
/// A object containing the
/// objects contained in the current
/// instance.
///
public IEnumerable Codecs {
get { return codecs; }
}
#endregion
#region ICodec
///
/// Gets the duration of the media represented by the current
/// instance.
///
///
/// A containing the duration of the
/// media represented by the current instance.
///
///
/// If the duration was set in the constructor, that value is
/// returned. Otherwise, the longest codec duration is used.
///
public TimeSpan Duration {
get {
TimeSpan duration = this.duration;
if (duration != TimeSpan.Zero)
return duration;
foreach (ICodec codec in codecs)
if (codec != null && codec.Duration > duration)
duration = codec.Duration;
return duration;
}
}
///
/// Gets the types of media represented by the current
/// instance.
///
///
/// A bitwise combined containing
/// the types of media represented by the current instance.
///
public MediaTypes MediaTypes {
get {
MediaTypes types = MediaTypes.None;
foreach (ICodec codec in codecs)
if (codec != null)
types |= codec.MediaTypes;
return types;
}
}
///
/// Gets a string description of the media represented by the
/// current instance.
///
///
/// A object containing a description
/// of the media represented by the current instance.
///
///
/// The value contains the descriptions of the codecs joined
/// by colons.
///
public string Description {
get {
var builder = new StringBuilder ();
foreach (ICodec codec in codecs) {
if (codec == null)
continue;
if (builder.Length != 0)
builder.Append ("; ");
builder.Append (codec.Description);
}
return builder.ToString ();
}
}
#endregion
#region IAudioCodec
///
/// Gets the bitrate of the audio represented by the current
/// instance.
///
///
/// A containing the bitrate of the audio
/// represented by the current instance.
///
///
/// This value is equal to the first non-zero audio bitrate.
///
public int AudioBitrate {
get {
foreach (ICodec codec in codecs) {
if (codec == null || (codec.MediaTypes & MediaTypes.Audio) == 0)
continue;
if (codec is IAudioCodec audio && audio.AudioBitrate != 0)
return audio.AudioBitrate;
}
return 0;
}
}
///
/// Gets the sample rate of the audio represented by the
/// current instance.
///
///
/// A containing the sample rate of the
/// audio represented by the current instance.
///
///
/// This value is equal to the first non-zero audio sample
/// rate.
///
public int AudioSampleRate {
get {
foreach (ICodec codec in codecs) {
if (codec == null ||
(codec.MediaTypes & MediaTypes.Audio) == 0)
continue;
if (codec is IAudioCodec audio && audio.AudioSampleRate != 0)
return audio.AudioSampleRate;
}
return 0;
}
}
///
/// Gets the number of bits per sample in the audio
/// represented by the current instance.
///
///
/// A value containing the number of bits
/// per sample in the audio represented by the current
/// instance.
///
///
/// This value is equal to the first non-zero quantization.
///
public int BitsPerSample {
get {
foreach (ICodec codec in codecs) {
if (codec == null || (codec.MediaTypes & MediaTypes.Audio) == 0)
continue;
if (codec is ILosslessAudioCodec lossless && lossless.BitsPerSample != 0)
return lossless.BitsPerSample;
}
return 0;
}
}
///
/// Gets the number of channels in the audio represented by
/// the current instance.
///
///
/// A object containing the number of
/// channels in the audio represented by the current
/// instance.
///
///
/// This value is equal to the first non-zero audio channel
/// count.
///
public int AudioChannels {
get {
foreach (ICodec codec in codecs) {
if (codec == null || (codec.MediaTypes & MediaTypes.Audio) == 0)
continue;
if (codec is IAudioCodec audio && audio.AudioChannels != 0)
return audio.AudioChannels;
}
return 0;
}
}
#endregion
#region IVideoCodec
///
/// Gets the width of the video represented by the current
/// instance.
///
///
/// A containing the width of the video
/// represented by the current instance.
///
///
/// This value is equal to the first non-zero video width.
///
public int VideoWidth {
get {
foreach (ICodec codec in codecs) {
if (codec == null || (codec.MediaTypes & MediaTypes.Video) == 0)
continue;
if (codec is IVideoCodec video && video.VideoWidth != 0)
return video.VideoWidth;
}
return 0;
}
}
///
/// Gets the height of the video represented by the current
/// instance.
///
///
/// A containing the height of the video
/// represented by the current instance.
///
///
/// This value is equal to the first non-zero video height.
///
public int VideoHeight {
get {
foreach (ICodec codec in codecs) {
if (codec == null || (codec.MediaTypes & MediaTypes.Video) == 0)
continue;
if (codec is IVideoCodec video && video.VideoHeight != 0)
return video.VideoHeight;
}
return 0;
}
}
#endregion
#region IPhotoCodec
///
/// Gets the width of the photo represented by the current
/// instance.
///
///
/// A value containing the width of the
/// photo represented by the current instance.
///
public int PhotoWidth {
get {
foreach (ICodec codec in codecs) {
if (codec == null || (codec.MediaTypes & MediaTypes.Photo) == 0)
continue;
if (codec is IPhotoCodec photo && photo.PhotoWidth != 0)
return photo.PhotoWidth;
}
return 0;
}
}
///
/// Gets the height of the photo represented by the current
/// instance.
///
///
/// A value containing the height of the
/// photo represented by the current instance.
///
public int PhotoHeight {
get {
foreach (ICodec codec in codecs) {
if (codec == null || (codec.MediaTypes & MediaTypes.Photo) == 0)
continue;
if (codec is IPhotoCodec photo && photo.PhotoHeight != 0)
return photo.PhotoHeight;
}
return 0;
}
}
///
/// Gets the (format specific) quality indicator of the photo
/// represented by the current instance.
///
///
/// A value indicating the quality. A value
/// 0 means that there was no quality indicator for the format
/// or the file.
///
public int PhotoQuality {
get {
foreach (ICodec codec in codecs) {
if (codec == null || (codec.MediaTypes & MediaTypes.Photo) == 0)
continue;
if (codec is IPhotoCodec photo && photo.PhotoQuality != 0)
return photo.PhotoQuality;
}
return 0;
}
}
#endregion
}
}