Tag.hxx 4.9 KB
Newer Older
Max Kellermann's avatar
Max Kellermann committed
1
/*
Max Kellermann's avatar
Max Kellermann committed
2
 * Copyright 2003-2017 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 "Compiler.h"
Max Kellermann's avatar
Max Kellermann committed
27 28

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

	/**
132 133
	 * 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
134 135
	 */
	gcc_pure
136
	const char *GetValue(TagType type) const noexcept;
Max Kellermann's avatar
Max Kellermann committed
137 138 139 140 141

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

	/**
	 * 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.
	 */
151
	gcc_pure gcc_returns_nonnull
152
	const char *GetSortValue(TagType type) const noexcept;
153 154 155 156 157

	class const_iterator {
		friend struct Tag;
		const TagItem *const*cursor;

158
		constexpr const_iterator(const TagItem *const*_cursor) noexcept
159 160 161
			:cursor(_cursor) {}

	public:
162
		constexpr const TagItem &operator*() const noexcept {
163 164 165
			return **cursor;
		}

166
		constexpr const TagItem *operator->() const noexcept {
167 168 169
			return *cursor;
		}

170
		const_iterator &operator++() noexcept {
171 172 173 174
			++cursor;
			return *this;
		}

175
		const_iterator operator++(int) noexcept {
176 177 178 179
			auto result = cursor++;
			return const_iterator{result};
		}

180
		const_iterator &operator--() noexcept {
181 182 183 184
			--cursor;
			return *this;
		}

185
		const_iterator operator--(int) noexcept {
186 187 188 189
			auto result = cursor--;
			return const_iterator{result};
		}

190
		constexpr bool operator==(const_iterator other) const noexcept {
191 192 193
			return cursor == other.cursor;
		}

194
		constexpr bool operator!=(const_iterator other) const noexcept {
195 196 197 198
			return cursor != other.cursor;
		}
	};

199
	const_iterator begin() const noexcept {
200 201 202
		return const_iterator{items};
	}

203
	const_iterator end() const noexcept {
204 205
		return const_iterator{items + num_items};
	}
Max Kellermann's avatar
Max Kellermann committed
206 207 208
};

#endif