cosmetix
[taglib.d.git] / taglib.d
blobafa320c80aea5f1dd953602ef74c195d41ae5111
1 ///WARING! this MAY BE 64-bit unsafe!
2 module taglib;
4 pragma(lib, "tag_c");
5 pragma(lib, "tag");
8 ////////////////////////////////////////////////////////////////////////////////
9 import std.conv;
10 import std.string;
13 class TagLibException : Exception {
14 this (string msg, string file=__FILE__, size_t line=__LINE__, Throwable next=null) @safe @pure @nothrow {
15 super(msg, file, line, next);
20 struct TagFile {
21 enum FileType {
22 Autodetect=-1, // k8 extension, do not use in C API!
23 MPEG,
24 OggVorbis,
25 FLAC,
26 MPC,
27 OggFlac,
28 WavPack,
29 Speex,
30 TrueAudio,
31 MP4,
32 ASF
36 this (string fname, FileType type=FileType.Autodetect) @trusted {
37 loadInfo(fname, type);
40 ~this () @trusted {
41 clear();
44 void clear () @trusted {
45 if (mInited) {
46 mInited = false;
47 taglib_file_free(mFL);
48 mFL = null;
49 mTags = null;
50 mFName = mArtist = mAlbum = mTitle = mComment = mGenre = null;
51 mYear = mTrack = 0;
55 void save () @trusted {
56 if (!mInited) throw new TagLibException("can't save tags object to empty file");
57 if (!taglib_file_save(mFL)) throw new TagLibException("can't save tags object to file '"~mFName~"'");
60 @property bool valid () const @trusted @nothrow { return mInited; }
62 @property string filename () const @trusted @nothrow { return (mInited ? mFName : null); }
64 mixin(strPropMixin("artist"));
65 mixin(strPropMixin("album"));
66 mixin(strPropMixin("title"));
67 mixin(strPropMixin("genre"));
68 mixin(strPropMixin("comment"));
70 @property uint year () const @trusted @nothrow { return (mInited ? mYear : 0); }
71 @property void year (uint v) @trusted {
72 if (!mInited) throw new TagLibException("can't set YEAR tag for empty file");
73 if (v > 0) {
74 uint ov;
75 if (v < 50) v += 2000;
76 else if (v < 100) v += 1900;
77 else if (v < 1930) v = 0;
78 if (v < 1930 || v > 2099) throw new TagLibException("invalid YEAR tag value: "~to!string(ov));
80 mYear = v;
81 taglib_tag_set_year(mTags, v);
84 @property uint track () const @trusted @nothrow { return (mInited ? mTrack : 0); }
85 @property void track (uint v) @trusted {
86 if (!mInited) throw new TagLibException("can't set TRACK tag for empty file");
87 if (v > 999) throw new TagLibException("invalid TRACK tag value: "~to!string(v));
88 mTrack = v;
89 taglib_tag_set_track(mTags, v);
92 mixin(uintPropMixin("length")); // file length in seconds
93 mixin(uintPropMixin("channels")); // number of channels in file
94 mixin(uintPropMixin("bitrate")); // file bitrate in kb/s
95 mixin(uintPropMixin("samplerate")); // file samplerate in Hz
97 private:
98 static string uintPropMixin (string propName) @pure @safe @nothrow {
99 return
100 `@property uint `~propName~` () @trusted {`~
101 `if (!mInited) return 0;`~
102 `auto tp = taglib_file_audioproperties(mFL);`~
103 `if (!tp) throw new TagLibException("can't get audio properties for file '"~mFName~"'");`~
104 `auto r = taglib_audioproperties_`~propName~`(tp);`~
105 `return (r < 0 ? 0 : r);`~
106 `}`;
109 static string strPropMixin (string propName) @pure @safe {
110 import std.string;
111 return
112 `@property void `~propName~` (string v) @trusted {`~
113 `if (!mInited) throw new TagLibException("can't set `~toUpper(propName)~` tag for empty file");`~
114 `string s = trimStr(v);`~
115 `taglib_tag_set_`~propName~`(mTags, s.toStringz);`~
116 `m`~capitalize(propName)~` = s.idup;`~
117 `}`~
118 `@property string `~propName~` () const @trusted @nothrow { return (mInited ? m`~capitalize(propName)~` : null); }`;
121 static string stripL (string str) @trusted {
122 import std.uni;
123 foreach (i, dchar c; str) if (c != '_' && !std.uni.isWhite(c)) return str[i..$];
124 return str[$..$];
127 static string stripR (string str) @trusted {
128 import std.uni, std.utf;
129 foreach_reverse (i, dchar c; str) if (c != '_' && !std.uni.isWhite(c)) return str[0..i+codeLength!char(c)];
130 return str[0..0];
133 static string trimStr (string s) @trusted {
134 import std.array;
135 auto res = appender!string();
136 dchar pch = 0;
137 foreach (dchar ch; s) {
138 if (ch < ' ') ch = ' ';
139 // remove duplicate underlines
140 if (pch == '_' && ch == '_') { pch = ch; continue; }
141 // remove duplicate spaces
142 if (pch == ' ' && ch == ' ') { pch = ch; continue; }
143 res.put(ch);
145 return stripL(stripR(res.data));
148 static string trimStr (char* s) @trusted {
149 if (s) {
150 auto res = trimStr(to!string(s));
151 taglib_free(s);
152 return res;
154 return "";
157 void loadInfo (string fname, FileType type=FileType.Autodetect) @trusted {
158 clear();
159 if (type == FileType.Autodetect) {
160 mFL = taglib_file_new(fname.toStringz);
161 } else {
162 mFL = taglib_file_new_type(fname.toStringz, cast(TagLibFileType)type);
164 scope(failure) clear();
165 if (!mFL) throw new TagLibException("can't open file '"~fname~"'");
166 mTags = taglib_file_tag(mFL);
167 if (!mTags) throw new TagLibException("can't init tags object for file '"~fname~"'");
168 mArtist = trimStr(taglib_tag_artist(mTags));
169 mAlbum = trimStr(taglib_tag_album(mTags));
170 mTitle = trimStr(taglib_tag_title(mTags));
171 mGenre = trimStr(taglib_tag_genre(mTags));
172 mYear = taglib_tag_year(mTags);
173 if (mYear > 0) {
174 if (mYear < 50) mYear += 2000;
175 else if (mYear < 100) mYear += 1900;
176 else if (mYear < 1930) mYear = 0;
178 if (mYear < 1930 || mYear > 2099) mYear = 0;
179 mTrack = taglib_tag_track(mTags);
180 if (mTrack > 999) mTrack = 0;
181 mFName = fname.idup;
182 mInited = true;
185 private:
186 bool mInited;
187 TagLibFile mFL;
188 TagLibTag mTags;
189 string mFName;
190 string mArtist;
191 string mAlbum;
192 string mTitle;
193 string mComment;
194 string mGenre;
195 uint mYear;
196 uint mTrack;
200 ////////////////////////////////////////////////////////////////////////////////
201 shared static this () {
202 taglib_set_strings_unicode(true);
203 taglib_set_string_management_enabled(false);
207 ////////////////////////////////////////////////////////////////////////////////
208 private:
209 extern(C):
210 @nothrow:
211 @trusted:
213 struct TagLibFileT {}
214 struct TagLibTagT {}
215 struct TagLibAudioPropertiesT {}
217 alias TagLibFile = TagLibFileT*;
218 alias TagLibTag = TagLibTagT*;
219 alias TagLibAudioProperties = TagLibAudioPropertiesT*;
220 alias TagBool = uint;
221 alias TagCString = const(char)*;
225 * By default all strings coming into or out of TagLib's C API are in UTF8.
226 * However, it may be desirable for TagLib to operate on Latin1 (ISO-8859-1)
227 * strings in which case this should be set to FALSE.
229 void taglib_set_strings_unicode (TagBool unicode);
232 * TagLib can keep track of strings that are created when outputting tag values
233 * and clear them using taglib_tag_clear_strings(). This is enabled by default.
234 * However if you wish to do more fine grained management of strings, you can do
235 * so by setting \a management to FALSE.
237 void taglib_set_string_management_enabled (TagBool management);
240 * Explicitly free a string returned from TagLib
242 void taglib_free (void* pointer);
245 /*******************************************************************************
246 * File API
247 ******************************************************************************/
248 enum TagLibFileType {
249 MPEG,
250 OggVorbis,
251 FLAC,
252 MPC,
253 OggFlac,
254 WavPack,
255 Speex,
256 TrueAudio,
257 MP4,
262 * Creates a TagLib file based on \a filename. TagLib will try to guess the file
263 * type.
265 * \returns NULL if the file type cannot be determined or the file cannot
266 * be opened.
268 TagLibFile taglib_file_new (TagCString filename);
271 * Creates a TagLib file based on \a filename. Rather than attempting to guess
272 * the type, it will use the one specified by \a type.
274 TagLibFile taglib_file_new_type (TagCString filename, TagLibFileType type);
277 * Frees and closes the file.
279 void taglib_file_free (TagLibFile file);
282 * Returns true if the file is open and readble and valid information for
283 * the Tag and / or AudioProperties was found.
286 TagBool taglib_file_is_valid (const(TagLibFile) file);
289 * Returns a pointer to the tag associated with this file. This will be freed
290 * automatically when the file is freed.
292 TagLibTag taglib_file_tag (const(TagLibFile) file);
295 * Returns a pointer to the the audio properties associated with this file. This
296 * will be freed automatically when the file is freed.
298 const(TagLibAudioProperties) taglib_file_audioproperties (const(TagLibFile) file);
301 * Saves the \a file to disk.
303 TagBool taglib_file_save (TagLibFile file);
306 /******************************************************************************
307 * Tag API
308 ******************************************************************************/
311 * Returns a string with this tag's title.
313 * \note By default this string should be UTF8 encoded and its memory should be
314 * freed using taglib_tag_free_strings().
316 char *taglib_tag_title (const(TagLibTag) tag);
319 * Returns a string with this tag's artist.
321 * \note By default this string should be UTF8 encoded and its memory should be
322 * freed using taglib_tag_free_strings().
324 char *taglib_tag_artist (const(TagLibTag) tag);
327 * Returns a string with this tag's album name.
329 * \note By default this string should be UTF8 encoded and its memory should be
330 * freed using taglib_tag_free_strings().
332 char *taglib_tag_album (const(TagLibTag) tag);
335 * Returns a string with this tag's comment.
337 * \note By default this string should be UTF8 encoded and its memory should be
338 * freed using taglib_tag_free_strings().
340 char *taglib_tag_comment (const(TagLibTag) tag);
343 * Returns a string with this tag's genre.
345 * \note By default this string should be UTF8 encoded and its memory should be
346 * freed using taglib_tag_free_strings().
348 char *taglib_tag_genre (const(TagLibTag) tag);
351 * Returns the tag's year or 0 if year is not set.
353 uint taglib_tag_year (const(TagLibTag) tag);
356 * Returns the tag's track number or 0 if track number is not set.
358 uint taglib_tag_track (const(TagLibTag) tag);
361 * Sets the tag's title.
363 * \note By default this string should be UTF8 encoded.
365 void taglib_tag_set_title (TagLibTag tag, TagCString title);
368 * Sets the tag's artist.
370 * \note By default this string should be UTF8 encoded.
372 void taglib_tag_set_artist (TagLibTag tag, TagCString artist);
375 * Sets the tag's album.
377 * \note By default this string should be UTF8 encoded.
379 void taglib_tag_set_album (TagLibTag tag, TagCString album);
382 * Sets the tag's comment.
384 * \note By default this string should be UTF8 encoded.
386 void taglib_tag_set_comment (TagLibTag tag, TagCString comment);
389 * Sets the tag's genre.
391 * \note By default this string should be UTF8 encoded.
393 void taglib_tag_set_genre (TagLibTag tag, TagCString genre);
396 * Sets the tag's year. 0 indicates that this field should be cleared.
398 void taglib_tag_set_year (TagLibTag tag, uint year);
401 * Sets the tag's track number. 0 indicates that this field should be cleared.
403 void taglib_tag_set_track (TagLibTag tag, uint track);
406 * Frees all of the strings that have been created by the tag.
408 void taglib_tag_free_strings ();
411 /******************************************************************************
412 * Audio Properties API
413 ******************************************************************************/
416 * Returns the length of the file in seconds.
418 int taglib_audioproperties_length (const(TagLibAudioProperties) audioProperties);
421 * Returns the bitrate of the file in kb/s.
423 int taglib_audioproperties_bitrate (const(TagLibAudioProperties) audioProperties);
426 * Returns the sample rate of the file in Hz.
428 int taglib_audioproperties_samplerate (const(TagLibAudioProperties) audioProperties);
431 * Returns the number of channels in the audio stream.
433 int taglib_audioproperties_channels (const(TagLibAudioProperties) audioProperties);
436 /*******************************************************************************
437 * Special convenience ID3v2 functions
438 *******************************************************************************/
439 enum TagLibID3v2Encoding {
440 Latin1,
441 UTF16,
442 UTF16BE,
443 UTF8
448 * This sets the default encoding for ID3v2 frames that are written to tags.
450 void taglib_id3v2_set_default_text_encoding (TagLibID3v2Encoding encoding);