lofty-rs/src/traits.rs

175 lines
4.7 KiB
Rust
Raw Normal View History

#[allow(clippy::wildcard_imports)]
2021-04-21 16:56:35 +00:00
use crate::components::tags::*;
use crate::{Album, AnyTag, Picture, Result, TagType};
2021-05-19 02:26:27 +00:00
use std::borrow::Cow;
use std::fs::{File, OpenOptions};
2020-10-27 14:38:31 +00:00
use lofty_attr::{i32_accessor, str_accessor, u16_accessor, u32_accessor};
/// Combination of [`AudioTagEdit`], [`AudioTagWrite`], and [`ToAnyTag`]
2020-10-29 18:01:21 +00:00
pub trait AudioTag: AudioTagEdit + AudioTagWrite + ToAnyTag {}
2020-10-27 14:38:31 +00:00
/// Implementors of this trait are able to read and write audio metadata.
///
/// Constructor methods e.g. `from_file` should be implemented separately.
pub trait AudioTagEdit {
str_accessor!(title);
str_accessor!(artist);
/// Splits the artist string into a `Vec`
fn artists(&self, delimiter: &str) -> Option<Vec<&str>> {
self.artist().map(|a| a.split(delimiter).collect())
}
i32_accessor!(year);
/// Returns the date
fn date(&self) -> Option<String> {
self.year().map(|y| y.to_string())
}
/// Sets the date
fn set_date(&mut self, _date: &str) {}
/// Removes the date
fn remove_date(&mut self) {}
str_accessor!(copyright);
str_accessor!(genre);
str_accessor!(lyrics);
u16_accessor!(bpm);
str_accessor!(lyricist);
str_accessor!(composer);
/// Returns the track's [`Album`]
fn album(&self) -> Album<'_> {
Album {
title: self.album_title(),
artist: self.album_artist(),
covers: self.album_covers(),
}
}
str_accessor!(album_title);
str_accessor!(album_artist);
/// Splits the artist string into a `Vec`
fn album_artists(&self, delimiter: &str) -> Option<Vec<&str>> {
self.album_artist().map(|a| a.split(delimiter).collect())
}
/// Returns the front and back album covers
fn album_covers(&self) -> (Option<Picture>, Option<Picture>) {
(self.front_cover(), self.back_cover())
}
/// Removes both album covers
fn remove_album_covers(&mut self) {
self.remove_front_cover();
self.remove_back_cover();
}
/// Returns the front cover
fn front_cover(&self) -> Option<Picture> {
None
}
/// Sets the front cover
fn set_front_cover(&mut self, _cover: Picture) {}
/// Removes the front cover
fn remove_front_cover(&mut self) {}
/// Returns the front cover
fn back_cover(&self) -> Option<Picture> {
None
}
/// Sets the front cover
fn set_back_cover(&mut self, _cover: Picture) {}
/// Removes the front cover
fn remove_back_cover(&mut self) {}
/// Returns an `Iterator` over all pictures stored in the track
fn pictures(&self) -> Option<Cow<'static, [Picture]>> {
None
}
/// Returns the track number and total tracks
fn track(&self) -> (Option<u32>, Option<u32>) {
(self.track_number(), self.total_tracks())
}
u32_accessor!(track_number);
u32_accessor!(total_tracks);
/// Returns the disc number and total discs
fn disc(&self) -> (Option<u32>, Option<u32>) {
(self.disc_number(), self.total_discs())
}
/// Removes the disc number and total discs
fn remove_disc(&mut self) {
self.remove_disc_number();
self.remove_total_discs();
}
u32_accessor!(disc_number);
u32_accessor!(total_discs);
2020-10-27 14:38:31 +00:00
}
/// Functions for writing to a file
2020-10-27 14:38:31 +00:00
pub trait AudioTagWrite {
/// Write tag to a [`File`][std::fs::File]
///
/// # Errors
///
/// Will return `Err` if unable to write to the `File`
fn write_to(&self, file: &mut File) -> Result<()>;
/// Write tag to a path
///
/// # Errors
///
/// Will return `Err` if `path` doesn't exist
fn write_to_path(&self, path: &str) -> Result<()> {
self.write_to(&mut OpenOptions::new().read(true).write(true).open(path)?)?;
Ok(())
}
2020-10-27 14:38:31 +00:00
}
/// Conversions between tag types
2020-10-29 18:01:21 +00:00
pub trait ToAnyTag: ToAny {
/// Converts the tag to [`AnyTag`]
fn to_anytag(&self) -> AnyTag<'_>;
/// Convert the tag type, which can be lossy.
fn to_dyn_tag(&self, tag_type: TagType) -> Box<dyn AudioTag> {
// TODO: write a macro or something that implement this method for every tag type so that if the
// TODO: target type is the same, just return self
match tag_type {
2021-05-17 03:07:26 +00:00
#[cfg(feature = "format-ape")]
TagType::Ape => Box::new(ApeTag::from(self.to_anytag())),
2021-05-17 03:07:26 +00:00
#[cfg(feature = "format-id3")]
2021-04-22 16:57:20 +00:00
TagType::Id3v2(_) => Box::new(Id3v2Tag::from(self.to_anytag())),
2021-05-17 03:07:26 +00:00
#[cfg(feature = "format-mp4")]
TagType::Mp4 => Box::new(Mp4Tag::from(self.to_anytag())),
2021-05-17 03:07:26 +00:00
#[cfg(any(
feature = "format-vorbis",
feature = "format-flac",
feature = "format-opus"
))]
TagType::Ogg(_) => Box::new(OggTag::from(self.to_anytag())),
2021-05-17 03:07:26 +00:00
#[cfg(feature = "format-riff")]
2021-04-22 16:57:20 +00:00
TagType::RiffInfo => Box::new(RiffTag::from(self.to_anytag())),
#[cfg(feature = "format-aiff")]
TagType::AiffText => Box::new(AiffTag::from(self.to_anytag())),
}
}
2020-10-27 14:38:31 +00:00
}
/// Tag conversion to `Any`
2020-10-29 18:01:21 +00:00
pub trait ToAny {
/// Convert tag to `Any`
fn to_any(&self) -> &dyn std::any::Any;
/// Mutably convert tag to `Any`
2021-04-23 02:57:47 +00:00
#[allow(clippy::wrong_self_convention)]
fn to_any_mut(&mut self) -> &mut dyn std::any::Any;
}