Tag.hxx 4.12 KB
Newer Older
Max Kellermann's avatar
Max Kellermann committed
1
/*
Max Kellermann's avatar
Max Kellermann committed
2
 * Copyright 2003-2020 The Music Player Daemon Project
Max Kellermann's avatar
Max Kellermann committed
3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22
 * http://www.musicpd.org
 *
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation; either version 2 of the License, or
 * (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License along
 * with this program; if not, write to the Free Software Foundation, Inc.,
 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
 */

#ifndef MPD_TAG_HXX
#define MPD_TAG_HXX

23 24
#include "Type.h" // IWYU pragma: export
#include "Item.hxx" // IWYU pragma: export
25
#include "Chrono.hxx"
26
#include "util/Compiler.h"
27
#include "util/DereferenceIterator.hxx"
Max Kellermann's avatar
Max Kellermann committed
28

29
#include <memory>
30
#include <utility>
Max Kellermann's avatar
Max Kellermann committed
31 32 33 34 35 36 37

/**
 * The meta information about a song file.  It is a MPD specific
 * subset of tags (e.g. from ID3, vorbis comments, ...).
 */
struct Tag {
	/**
38 39
	 * The duration of the song.  A negative value means that the
	 * length is unknown.
Max Kellermann's avatar
Max Kellermann committed
40
	 */
41
	SignedSongTime duration = SignedSongTime::Negative();
Max Kellermann's avatar
Max Kellermann committed
42 43 44 45 46

	/**
	 * Does this file have an embedded playlist (e.g. embedded CUE
	 * sheet)?
	 */
47
	bool has_playlist = false;
Max Kellermann's avatar
Max Kellermann committed
48

49
	/** the total number of tag items in the #items array */
50
	unsigned short num_items = 0;
51

Max Kellermann's avatar
Max Kellermann committed
52
	/** an array of tag items */
53
	TagItem **items = nullptr;
Max Kellermann's avatar
Max Kellermann committed
54 55 56 57

	/**
	 * Create an empty tag.
	 */
58
	Tag() = default;
Max Kellermann's avatar
Max Kellermann committed
59

60
	Tag(const Tag &other) noexcept;
Max Kellermann's avatar
Max Kellermann committed
61

62
	Tag(Tag &&other) noexcept
63
		:duration(other.duration), has_playlist(other.has_playlist),
64
		 num_items(other.num_items), items(other.items) {
Max Kellermann's avatar
Max Kellermann committed
65 66 67 68 69 70 71
		other.items = nullptr;
		other.num_items = 0;
	}

	/**
	 * Free the tag object and all its items.
	 */
72
	~Tag() noexcept {
73 74
		Clear();
	}
Max Kellermann's avatar
Max Kellermann committed
75 76 77

	Tag &operator=(const Tag &other) = delete;

78
	Tag &operator=(Tag &&other) noexcept {
79
		duration = other.duration;
Max Kellermann's avatar
Max Kellermann committed
80
		has_playlist = other.has_playlist;
81 82 83 84 85 86 87 88
		MoveItemsFrom(std::move(other));
		return *this;
	}

	/**
	 * Similar to the move operator, but move only the #TagItem
	 * array.
	 */
89
	void MoveItemsFrom(Tag &&other) noexcept {
Max Kellermann's avatar
Max Kellermann committed
90 91 92 93 94
		std::swap(items, other.items);
		std::swap(num_items, other.num_items);
	}

	/**
95 96
	 * Returns true if the tag contains no items.  This ignores
	 * the "duration" attribute.
Max Kellermann's avatar
Max Kellermann committed
97
	 */
98
	bool IsEmpty() const noexcept {
Max Kellermann's avatar
Max Kellermann committed
99 100 101 102 103 104
		return num_items == 0;
	}

	/**
	 * Returns true if the tag contains any information.
	 */
105
	bool IsDefined() const noexcept {
106
		return !IsEmpty() || !duration.IsNegative();
Max Kellermann's avatar
Max Kellermann committed
107 108
	}

Max Kellermann's avatar
Max Kellermann committed
109 110 111
	/**
	 * Clear everything, as if this was a new Tag object.
	 */
112
	void Clear() noexcept;
Max Kellermann's avatar
Max Kellermann committed
113

Max Kellermann's avatar
Max Kellermann committed
114 115
	/**
	 * Merges the data from two tags.  If both tags share data for the
116
	 * same TagType, only data from "add" is used.
Max Kellermann's avatar
Max Kellermann committed
117 118 119
	 *
	 * @return a newly allocated tag
	 */
120 121
	static std::unique_ptr<Tag> Merge(const Tag &base,
					  const Tag &add) noexcept;
Max Kellermann's avatar
Max Kellermann committed
122 123

	/**
124
	 * Merges the data from two tags.  Any of the two may be nullptr.  Both
Max Kellermann's avatar
Max Kellermann committed
125 126 127 128
	 * are freed by this function.
	 *
	 * @return a newly allocated tag
	 */
129
	static std::unique_ptr<Tag> Merge(std::unique_ptr<Tag> base,
130
					  std::unique_ptr<Tag> add) noexcept;
Max Kellermann's avatar
Max Kellermann committed
131 132

	/**
133 134
	 * Returns the first value of the specified tag type, or
	 * nullptr if none is present in this tag object.
Max Kellermann's avatar
Max Kellermann committed
135 136
	 */
	gcc_pure
137
	const char *GetValue(TagType type) const noexcept;
Max Kellermann's avatar
Max Kellermann committed
138 139 140 141 142

	/**
	 * Checks whether the tag contains one or more items with
	 * the specified type.
	 */
143
	gcc_pure
144
	bool HasType(TagType type) const noexcept;
145 146 147 148 149 150 151

	/**
	 * Returns a value for sorting on the specified type, with
	 * automatic fallbacks to the next best tag type
	 * (e.g. #TAG_ALBUM_ARTIST falls back to #TAG_ARTIST).  If
	 * there is no such value, returns an empty string.
	 */
152
	gcc_pure gcc_returns_nonnull
153
	const char *GetSortValue(TagType type) const noexcept;
154

155 156
	using const_iterator = DereferenceIterator<TagItem *const*,
						   const TagItem>;
157

158
	const_iterator begin() const noexcept {
159 160 161
		return const_iterator{items};
	}

162
	const_iterator end() const noexcept {
163 164
		return const_iterator{items + num_items};
	}
Max Kellermann's avatar
Max Kellermann committed
165 166 167
};

#endif