mirror of
				https://github.com/RGBCube/serenity
				synced 2025-10-25 02:12:32 +00:00 
			
		
		
		
	 d8e8ddedf3
			
		
	
	
		d8e8ddedf3
		
	
	
	
	
		
			
			This container has several design goals: - Represent all common and relevant metadata fields of audio files in a unified way. - Allow perfect recreation of any metadata format from the in-memory structure. This requires that we allow non-detected fields to reside in an "untyped" miscellaneous collection. Like with pictures, plugins are free to store their metadata into the m_metadata field whenever they read it. It is recommended that this happens on loader creation; however failing to read metadata should not cause an error in the plugin.
		
			
				
	
	
		
			69 lines
		
	
	
	
		
			2.1 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			69 lines
		
	
	
	
		
			2.1 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|  * Copyright (c) 2023, kleines Filmröllchen <filmroellchen@serenityos.org>
 | |
|  *
 | |
|  * SPDX-License-Identifier: BSD-2-Clause
 | |
|  */
 | |
| 
 | |
| #pragma once
 | |
| 
 | |
| #include <AK/HashMap.h>
 | |
| #include <AK/LexicalPath.h>
 | |
| #include <AK/Optional.h>
 | |
| #include <AK/String.h>
 | |
| #include <AK/Time.h>
 | |
| #include <AK/Variant.h>
 | |
| 
 | |
| namespace Audio {
 | |
| 
 | |
| struct Person {
 | |
|     enum class Role {
 | |
|         Artist,
 | |
|         Performer,
 | |
|         Lyricist,
 | |
|         Conductor,
 | |
|         Publisher,
 | |
|         Engineer,
 | |
|         Composer,
 | |
|     };
 | |
|     Role role;
 | |
|     String name;
 | |
| 
 | |
|     // Whether this person has creative involvement with the song (so not only Role::Artist!).
 | |
|     // This list is subjective and is intended to keep the artist display text in applications relevant.
 | |
|     // It is used for first_artist and all_artists in Metadata.
 | |
|     bool is_artist() const;
 | |
| 
 | |
|     Optional<StringView> name_for_role() const;
 | |
| };
 | |
| 
 | |
| // Audio metadata of the original format must be equivalently reconstructible from this struct.
 | |
| // That means, (if the format allows it) fields can appear in a different order, but all fields must be present with the original values,
 | |
| // including duplicate fields where allowed by the format.
 | |
| struct Metadata {
 | |
|     using Year = unsigned;
 | |
| 
 | |
|     void replace_encoder_with_serenity();
 | |
|     ErrorOr<void> add_miscellaneous(String const& field, String value);
 | |
|     ErrorOr<void> add_person(Person::Role role, String name);
 | |
|     Optional<String> first_artist() const;
 | |
|     ErrorOr<Optional<String>> all_artists(StringView concatenate_with = ", "sv) const;
 | |
| 
 | |
|     Optional<String> title;
 | |
|     Optional<String> subtitle;
 | |
|     Optional<unsigned> track_number;
 | |
|     Optional<String> album;
 | |
|     Optional<String> genre;
 | |
|     Optional<String> comment;
 | |
|     Optional<String> isrc;
 | |
|     Optional<String> encoder;
 | |
|     Optional<String> copyright;
 | |
|     Optional<float> bpm;
 | |
|     // FIXME: Until the time data structure situation is solved in a good way, we don't parse ISO 8601 time specifications.
 | |
|     Optional<String> unparsed_time;
 | |
|     Vector<Person> people;
 | |
| 
 | |
|     // Any other metadata, using the format-specific field names. This ensures reproducibility.
 | |
|     HashMap<String, Vector<String>> miscellaneous;
 | |
| };
 | |
| 
 | |
| }
 |