aboutsummaryrefslogtreecommitdiff
path: root/Src/replicant/nsid3v2/frame_apic.cpp
diff options
context:
space:
mode:
authorJean-Francois Mauguit <jfmauguit@mac.com>2024-09-24 09:03:25 -0400
committerGitHub <noreply@github.com>2024-09-24 09:03:25 -0400
commitbab614c421ed7ae329d26bf028c4a3b1d2450f5a (patch)
tree12f17f78986871dd2cfb0a56e5e93b545c1ae0d0 /Src/replicant/nsid3v2/frame_apic.cpp
parent4bde6044fddf053f31795b9eaccdd2a5a527d21f (diff)
parent20d28e80a5c861a9d5f449ea911ab75b4f37ad0d (diff)
downloadwinamp-bab614c421ed7ae329d26bf028c4a3b1d2450f5a.tar.gz
Merge pull request #5 from WinampDesktop/community
Merge to main
Diffstat (limited to 'Src/replicant/nsid3v2/frame_apic.cpp')
-rw-r--r--Src/replicant/nsid3v2/frame_apic.cpp247
1 files changed, 247 insertions, 0 deletions
diff --git a/Src/replicant/nsid3v2/frame_apic.cpp b/Src/replicant/nsid3v2/frame_apic.cpp
new file mode 100644
index 00000000..1276c29d
--- /dev/null
+++ b/Src/replicant/nsid3v2/frame_apic.cpp
@@ -0,0 +1,247 @@
+#include "nsid3v2.h"
+#include "nsid3v2/header.h"
+#include "nsid3v2/tag.h"
+#include "nsid3v2/frame_utils.h"
+#include "nu/ByteReader.h"
+#include "nu/ByteWriter.h"
+#include "nx/nxstring.h"
+
+struct ParsedPicture
+{
+ ParsedString mime;
+ uint8_t picture_type;
+ ParsedString description;
+ const void *picture_data;
+ size_t picture_byte_length;
+};
+
+static int ParsePicture(const void *data, size_t data_len, ParsedPicture &parsed)
+{
+ int ret;
+ if (data_len < 4)
+ return NErr_NeedMoreData;
+
+ bytereader_value_t byte_reader;
+ bytereader_init(&byte_reader, data, data_len);
+
+ uint8_t encoding = bytereader_read_u8(&byte_reader);
+ /* mime type is always latin-1 */
+ ret = ParseNullTerminatedString(&byte_reader, 0, parsed.mime);
+ if (ret != NErr_Success)
+ return ret;
+
+ if (bytereader_size(&byte_reader) < 2)
+ return NErr_NeedMoreData;
+
+ parsed.picture_type = bytereader_read_u8(&byte_reader);
+
+ ret = ParseNullTerminatedString(&byte_reader, encoding, parsed.description);
+ if (ret != NErr_Success)
+ return ret;
+
+ parsed.picture_data = bytereader_pointer(&byte_reader);
+ parsed.picture_byte_length = bytereader_size(&byte_reader);
+ return NErr_Success;
+}
+
+static int ParsePicturev2_2(const void *data, size_t data_len, ParsedPicture &parsed)
+{
+ int ret;
+ if (data_len < 6)
+ return NErr_NeedMoreData;
+
+ bytereader_value_t byte_reader;
+ bytereader_init(&byte_reader, data, data_len);
+
+ uint8_t encoding = bytereader_read_u8(&byte_reader);
+
+ /* three byte "Image Format" field */
+ parsed.mime.encoding = 0;
+ parsed.mime.data = bytereader_pointer(&byte_reader);
+ parsed.mime.byte_length = 3;
+
+ bytereader_advance(&byte_reader, 3);
+
+ parsed.picture_type = bytereader_read_u8(&byte_reader);
+
+ ret = ParseNullTerminatedString(&byte_reader, encoding, parsed.description);
+ if (ret != NErr_Success)
+ return ret;
+
+ parsed.picture_data = bytereader_pointer(&byte_reader);
+ parsed.picture_byte_length = bytereader_size(&byte_reader);
+ return NErr_Success;
+}
+
+int NSID3v2_Frame_Picture_Get(const nsid3v2_frame_t f, nx_string_t *mime, uint8_t *picture_type, nx_string_t *description, const void **picture_data, size_t *length, int text_flags)
+{
+ const ID3v2::Frame *frame = (const ID3v2::Frame *)f;
+ if (frame)
+ {
+ const void *data;
+ size_t data_len;
+ ParsedPicture parsed;
+ if (frame->GetData(&data, &data_len) == NErr_Success && data_len > 0)
+ {
+ int ret;
+ if (frame->GetVersion() == 2)
+ ret = ParsePicturev2_2(data, data_len, parsed);
+ else
+ ret = ParsePicture(data, data_len, parsed);
+
+ if (ret == NErr_Success)
+ {
+ int ret;
+ if (mime)
+ {
+ ret = NXStringCreateFromParsedString(mime, parsed.mime, text_flags);
+ if (ret != NErr_Success)
+ return ret;
+ }
+
+ if (description)
+ {
+ ret = NXStringCreateFromParsedString(description, parsed.description, text_flags);
+ if (ret != NErr_Success)
+ return ret;
+ }
+
+ if (picture_type)
+ *picture_type = parsed.picture_type;
+
+ if (picture_data)
+ *picture_data = parsed.picture_data;
+ if (length)
+ *length = parsed.picture_byte_length;
+
+ return NErr_Success;
+ }
+ else
+ {
+ return ret;
+ }
+ }
+
+ }
+ return NErr_Empty;
+}
+
+/* ---------------- Setters ---------------- */
+static const char *GetMIME2_2(nx_string_t mime)
+{
+ if (!mime)
+ return "\0\0\0";
+
+ if (NXStringKeywordCompareWithCString(mime, "image/jpeg") == NErr_True || NXStringKeywordCompareWithCString(mime, "image/jpg") == NErr_True)
+ return "JPG";
+
+ if (NXStringKeywordCompareWithCString(mime, "image/png") == NErr_True)
+ return "PNG";
+
+ if (NXStringKeywordCompareWithCString(mime, "image/gif") == NErr_True)
+ return "GIF";
+
+ if (NXStringKeywordCompareWithCString(mime, "image/bmp") == NErr_True)
+ return "BMP";
+
+ return "\0\0\0";
+}
+
+int NSID3v2_Frame_Picture_Set(nsid3v2_frame_t f, nx_string_t mime, uint8_t picture_type, nx_string_t description, const void *picture_data, size_t length, int text_flags)
+{
+ ID3v2::Frame *frame = (ID3v2::Frame *)f;
+ if (frame)
+ {
+ if (frame->GetVersion() == 2)
+ {
+ /* first, we need to get the total encoded size */
+
+ size_t byte_count_description=0;
+ if (description)
+ {
+ int ret = NXStringGetBytesSize(&byte_count_description, description, nx_charset_latin1, 0);
+ if (ret != NErr_DirectPointer && ret != NErr_Success)
+ return ret;
+ }
+
+ size_t total_size = 1 /* text encoding */
+ + 3 /* Image Format is 3 bytes in ID3v2.2*/
+ + 1 /* picture type */
+ + byte_count_description + 1 /* description + null terminator */
+ + length; /* picture length */
+
+ void *data;
+ size_t data_size;
+ int ret = frame->NewData(total_size, &data, &data_size);
+ if (ret != NErr_Success)
+ return ret;
+
+ size_t bytes_copied;
+ bytewriter_s byte_writer;
+ bytewriter_init(&byte_writer, data, data_size);
+ bytewriter_write_u8(&byte_writer, 0); /* mark as Latin-1 */
+ bytewriter_write_n(&byte_writer, GetMIME2_2(mime), 3);
+ bytewriter_write_u8(&byte_writer, picture_type);
+ if (description)
+ {
+ NXStringGetBytes(&bytes_copied, description, bytewriter_pointer(&byte_writer), bytewriter_size(&byte_writer), nx_charset_latin1, 0);
+ bytewriter_advance(&byte_writer, bytes_copied);
+ }
+ bytewriter_write_u8(&byte_writer, 0); /* description null terminator */
+ bytewriter_write_n(&byte_writer, picture_data, length);
+ return NErr_Success;
+ }
+ else
+ {
+ /* first, we need to get the total encoded size */
+ size_t byte_count_mime=0;
+ if (mime)
+ {
+ int ret = NXStringGetBytesSize(&byte_count_mime, mime, nx_charset_latin1, 0);
+ if (ret != NErr_DirectPointer && ret != NErr_Success)
+ return ret;
+ }
+
+ size_t byte_count_description=0;
+ if (description)
+ {
+ int ret = NXStringGetBytesSize(&byte_count_description, description, nx_charset_latin1, 0);
+ if (ret != NErr_DirectPointer && ret != NErr_Success)
+ return ret;
+ }
+
+ size_t total_size = 1 /* text encoding */
+ + byte_count_mime + 1 /* mime + null terminator */
+ + 1 /* picture type */
+ + byte_count_description + 1 /* description + null terminator */
+ + length; /* picture length */
+
+ void *data;
+ size_t data_size;
+ int ret = frame->NewData(total_size, &data, &data_size);
+ if (ret != NErr_Success)
+ return ret;
+
+ size_t bytes_copied;
+ bytewriter_s byte_writer;
+ bytewriter_init(&byte_writer, data, data_size);
+ bytewriter_write_u8(&byte_writer, 0); /* mark as Latin-1 */
+ if (mime)
+ {
+ NXStringGetBytes(&bytes_copied, mime, bytewriter_pointer(&byte_writer), bytewriter_size(&byte_writer), nx_charset_latin1, 0);
+ bytewriter_advance(&byte_writer, bytes_copied);
+ }
+ bytewriter_write_u8(&byte_writer, 0); /* MIME null terminator */
+ bytewriter_write_u8(&byte_writer, picture_type);
+ if (description)
+ {
+ NXStringGetBytes(&bytes_copied, description, bytewriter_pointer(&byte_writer), bytewriter_size(&byte_writer), nx_charset_latin1, 0);
+ bytewriter_advance(&byte_writer, bytes_copied);
+ }
+ bytewriter_write_u8(&byte_writer, 0); /* description null terminator */
+ bytewriter_write_n(&byte_writer, picture_data, length);
+ return NErr_Success;
+ }
+ }
+ return NErr_Empty;
+}