diff options
author | Jean-Francois Mauguit <jfmauguit@mac.com> | 2024-09-24 09:03:25 -0400 |
---|---|---|
committer | GitHub <noreply@github.com> | 2024-09-24 09:03:25 -0400 |
commit | bab614c421ed7ae329d26bf028c4a3b1d2450f5a (patch) | |
tree | 12f17f78986871dd2cfb0a56e5e93b545c1ae0d0 /Src/replicant/nsid3v2/frame_popm.cpp | |
parent | 4bde6044fddf053f31795b9eaccdd2a5a527d21f (diff) | |
parent | 20d28e80a5c861a9d5f449ea911ab75b4f37ad0d (diff) | |
download | winamp-bab614c421ed7ae329d26bf028c4a3b1d2450f5a.tar.gz |
Merge pull request #5 from WinampDesktop/community
Merge to main
Diffstat (limited to 'Src/replicant/nsid3v2/frame_popm.cpp')
-rw-r--r-- | Src/replicant/nsid3v2/frame_popm.cpp | 95 |
1 files changed, 95 insertions, 0 deletions
diff --git a/Src/replicant/nsid3v2/frame_popm.cpp b/Src/replicant/nsid3v2/frame_popm.cpp new file mode 100644 index 00000000..e2a4f6a5 --- /dev/null +++ b/Src/replicant/nsid3v2/frame_popm.cpp @@ -0,0 +1,95 @@ +#include "nsid3v2.h" +#include "nsid3v2/header.h" +#include "nsid3v2/tag.h" +#include "nsid3v2/frame_utils.h" +#include "nu/ByteReader.h" +#include "nx/nxstring.h" +#if defined(_WIN32) && !defined(strcasecmp) +#define strcasecmp _stricmp +#else +#include <strings.h> +#endif + +struct ParsedPopularimeter +{ + ParsedString email; + uint8_t rating; + uint64_t playcount; +}; + +static int ParsePopularimeter(const void *data, size_t data_len, ParsedPopularimeter &parsed) +{ + int ret; + if (data_len < 6) + return NErr_Insufficient; + + bytereader_value_t byte_reader; + bytereader_init(&byte_reader, data, data_len); + + /* read email (Always latin-1) */ + ret = ParseNullTerminatedString(&byte_reader, 0, parsed.email); + if (ret != NErr_Success) + return ret; + + if (bytereader_size(&byte_reader) == 0) + return NErr_Insufficient; + + parsed.rating = bytereader_read_u8(&byte_reader); + + parsed.playcount=0; + while (bytereader_size(&byte_reader)) + { + + parsed.playcount <<= 8; + parsed.playcount |= bytereader_read_u8(&byte_reader); + } + return NErr_Success; +} + +int NSID3v2_Tag_Popularimeter_GetRatingPlaycount(const nsid3v2_tag_t t, const char *email, uint8_t *rating, uint64_t *playcount) +{ + const ID3v2::Tag *tag = (const ID3v2::Tag *)t; + if (!tag) + return NErr_Empty; + const ID3v2::Frame *frame = tag->FindFirstFrame(NSID3V2_FRAME_POPULARIMETER); + while (frame) + { + const void *data; + size_t data_len; + ParsedPopularimeter parsed; + if (frame->GetData(&data, &data_len) == NErr_Success && data_len > 0 && ParsePopularimeter(data, data_len, parsed) == NErr_Success) + { + if (!strcasecmp(email, (const char *)parsed.email.data)) + { + *rating = parsed.rating; + *playcount = parsed.playcount; + return NErr_Success; + } + + } + frame = tag->FindNextFrame(frame); + } + + return NErr_Empty; +} + +int NSID3v2_Frame_Popularity_Get(nsid3v2_frame_t f, nx_string_t *email, uint8_t *rating, uint64_t *playcount, int text_flags) +{ + const ID3v2::Frame *frame = (const ID3v2::Frame *)f; + if (frame) + { + const void *data; + size_t data_len; + ParsedPopularimeter parsed; + if (frame->GetData(&data, &data_len) == NErr_Success && data_len > 0 && ParsePopularimeter(data, data_len, parsed) == NErr_Success) + { + int ret = NXStringCreateFromParsedString(email, parsed.email, text_flags); + if (ret != NErr_Success) + return ret; + *rating = parsed.rating; + *playcount = parsed.playcount; + return NErr_Success; + } + } + return NErr_Empty; +} |