// // OMPTMetadataReader.m // OpenMPT // // Created by Christopher Snowhill on 1/4/18. // Copyright 2018 __LoSnoCo__. All rights reserved. // #import "OMPTMetadataReader.h" #import "OMPTDecoder.h" #import #import "Logging.H" @implementation OMPTMetadataReader + (NSArray *)fileTypes { return [OMPTDecoder fileTypes]; } + (NSArray *)mimeTypes { return [OMPTDecoder mimeTypes]; } + (float)priority { return 1.0f; } static void setDictionary(NSMutableDictionary *dict, NSString *tag, NSString *value) { NSMutableArray *array = [dict valueForKey:tag]; if(!array) { array = [[NSMutableArray alloc] init]; [dict setObject:array forKey:tag]; } [array addObject:value]; } + (NSDictionary *)metadataForURL:(NSURL *)url { id audioSourceClass = NSClassFromString(@"AudioSource"); id source = [audioSourceClass audioSourceForURL:url]; if(![source open:url]) return 0; if(![source seekable]) return 0; [source seek:0 whence:SEEK_END]; long size = [source tell]; [source seek:0 whence:SEEK_SET]; std::vector data(static_cast(size)); [source read:data.data() amount:size]; int track_num; if([[url fragment] length] == 0) track_num = 0; else track_num = [[url fragment] intValue]; try { std::map ctls; openmpt::module *mod = new openmpt::module(data, std::clog, ctls); NSMutableDictionary *dict = [[NSMutableDictionary alloc] init]; std::vector keys = mod->get_metadata_keys(); for(std::vector::iterator key = keys.begin(); key != keys.end(); ++key) { NSString *tag = guess_encoding_of_string((*key).c_str()); NSString *value = guess_encoding_of_string(mod->get_metadata(*key).c_str()); if(*key == "type") continue; else if(*key == "type_long") { setDictionary(dict, @"codec", value); } else { setDictionary(dict, tag, value); } } delete mod; return dict; } catch(std::exception & /*e*/) { return 0; } } @end