2015-12-10 13:59:21 -05:00
|
|
|
/*
|
2013-12-22 17:24:09 -05:00
|
|
|
* Copyright 2013 (c) Anna Schumaker.
|
|
|
|
*/
|
2015-12-07 09:35:58 -05:00
|
|
|
#include <core/collection.h>
|
2015-12-09 09:55:55 -05:00
|
|
|
#include <core/string.h>
|
2014-06-05 10:19:22 -04:00
|
|
|
#include <core/playlist.h>
|
2013-12-22 17:24:09 -05:00
|
|
|
|
2015-11-04 12:23:00 -05:00
|
|
|
static struct database playlist_db;
|
2015-12-10 10:55:53 -05:00
|
|
|
static struct queue playlist_q;
|
2015-12-10 13:26:22 -05:00
|
|
|
static enum playlist_t playlist_cur;
|
|
|
|
static const gchar *playlist_names[2] = { "Favorites", "Banned" };
|
2014-02-02 14:20:56 -05:00
|
|
|
|
2015-12-10 10:55:53 -05:00
|
|
|
|
2015-12-10 13:59:21 -05:00
|
|
|
static inline struct index_entry *__playlist_lookup(enum playlist_t plist)
|
2015-12-09 09:55:55 -05:00
|
|
|
{
|
2015-12-10 13:26:22 -05:00
|
|
|
return INDEX_ENTRY(db_get(&playlist_db, playlist_names[plist]));
|
2015-12-09 09:55:55 -05:00
|
|
|
}
|
|
|
|
|
2015-12-10 13:26:22 -05:00
|
|
|
static inline bool __playlist_is_static(enum playlist_t plist)
|
2015-12-09 09:55:55 -05:00
|
|
|
{
|
2016-02-26 15:37:27 -05:00
|
|
|
return (plist == PL_FAVORITED) || (plist == PL_HIDDEN);
|
2015-12-10 13:26:22 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
static inline bool __playlist_is_dynamic(enum playlist_t plist)
|
|
|
|
{
|
|
|
|
return !__playlist_is_static(plist);
|
2015-12-09 09:55:55 -05:00
|
|
|
}
|
|
|
|
|
2015-12-10 10:55:53 -05:00
|
|
|
static unsigned int __playlist_find_average()
|
|
|
|
{
|
|
|
|
struct db_entry *track, *next;
|
|
|
|
unsigned int total = 0;
|
|
|
|
|
|
|
|
if (track_db_get()->db_size == 0)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
db_for_each(track, next, track_db_get())
|
|
|
|
total += TRACK(track)->tr_count;
|
|
|
|
return total / track_db_get()->db_size;
|
|
|
|
}
|
|
|
|
|
2015-12-10 13:26:22 -05:00
|
|
|
static void __playlist_fill_static(enum playlist_t plist)
|
2015-12-10 10:55:53 -05:00
|
|
|
{
|
2015-12-10 13:26:22 -05:00
|
|
|
struct index_entry *ent = __playlist_lookup(plist);
|
2015-12-10 13:38:30 -05:00
|
|
|
struct set_iter it;
|
|
|
|
|
|
|
|
if (ent) {
|
|
|
|
set_for_each(&ent->ie_set, &it)
|
|
|
|
queue_add(&playlist_q, track_get(it.it_val));
|
|
|
|
}
|
2015-12-10 10:55:53 -05:00
|
|
|
}
|
|
|
|
|
2015-12-10 13:26:22 -05:00
|
|
|
static void __playlist_fill_dynamic(enum playlist_t plist)
|
2015-12-10 10:55:53 -05:00
|
|
|
{
|
|
|
|
unsigned int count = 0, average = 0;
|
2015-12-10 13:26:22 -05:00
|
|
|
struct db_entry *track, *next;
|
2015-12-10 10:55:53 -05:00
|
|
|
|
2015-12-10 13:26:22 -05:00
|
|
|
if (plist == PL_MOST_PLAYED || plist == PL_LEAST_PLAYED)
|
2015-12-10 10:55:53 -05:00
|
|
|
average = __playlist_find_average();
|
|
|
|
|
|
|
|
db_for_each(track, next, track_db_get()) {
|
|
|
|
count = TRACK(track)->tr_count;
|
2015-12-10 13:26:22 -05:00
|
|
|
if (plist == PL_UNPLAYED && count == 0)
|
2015-12-10 13:38:30 -05:00
|
|
|
queue_add(&playlist_q, TRACK(track));
|
2015-12-10 13:26:22 -05:00
|
|
|
else if (plist == PL_MOST_PLAYED && count > average)
|
2015-12-10 13:38:30 -05:00
|
|
|
queue_add(&playlist_q, TRACK(track));
|
2015-12-10 13:26:22 -05:00
|
|
|
else if (plist == PL_LEAST_PLAYED && count < average && count > 0)
|
2015-12-10 13:38:30 -05:00
|
|
|
queue_add(&playlist_q, TRACK(track));
|
2015-12-10 10:55:53 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-05-18 21:09:46 -04:00
|
|
|
|
2015-12-09 08:11:21 -05:00
|
|
|
void playlist_init(struct queue_ops *ops)
|
2014-05-18 21:09:46 -04:00
|
|
|
{
|
2015-11-29 19:45:39 -05:00
|
|
|
queue_init(&playlist_q, Q_ENABLED | Q_REPEAT, ops);
|
2015-12-03 13:43:15 -05:00
|
|
|
queue_sort(&playlist_q, COMPARE_ARTIST, true);
|
|
|
|
queue_sort(&playlist_q, COMPARE_YEAR, false);
|
|
|
|
queue_sort(&playlist_q, COMPARE_TRACK, false);
|
2015-12-03 13:41:39 -05:00
|
|
|
queue_set_flag(&playlist_q, Q_NO_SORT);
|
2015-11-23 08:36:29 -05:00
|
|
|
|
2015-09-29 16:25:16 -04:00
|
|
|
index_init(&playlist_db, "playlist.db", true);
|
2015-09-17 08:12:25 -04:00
|
|
|
db_load(&playlist_db);
|
2014-05-18 21:09:46 -04:00
|
|
|
}
|
2014-02-02 14:11:38 -05:00
|
|
|
|
2015-12-09 09:20:18 -05:00
|
|
|
void playlist_deinit()
|
|
|
|
{
|
|
|
|
queue_deinit(&playlist_q);
|
|
|
|
db_deinit(&playlist_db);
|
|
|
|
}
|
|
|
|
|
2015-12-10 14:44:31 -05:00
|
|
|
bool playlist_add(enum playlist_t plist, struct track *track)
|
2013-12-22 17:53:58 -05:00
|
|
|
{
|
2015-12-10 14:44:31 -05:00
|
|
|
if (!track || !__playlist_is_static(plist) || playlist_has(plist, track))
|
|
|
|
return false;
|
2014-05-18 21:09:46 -04:00
|
|
|
|
2015-12-10 14:44:31 -05:00
|
|
|
index_insert(&playlist_db, playlist_names[plist],
|
|
|
|
track->tr_dbe.dbe_index);
|
|
|
|
if (playlist_cur == plist)
|
|
|
|
queue_add(&playlist_q, track);
|
|
|
|
return true;
|
2013-12-22 17:53:58 -05:00
|
|
|
}
|
|
|
|
|
2015-12-11 08:21:21 -05:00
|
|
|
bool playlist_remove(enum playlist_t plist, struct track *track)
|
2013-12-22 17:53:58 -05:00
|
|
|
{
|
2015-12-11 08:21:21 -05:00
|
|
|
if (!track || !playlist_has(plist, track))
|
|
|
|
return false;
|
2015-12-10 09:20:42 -05:00
|
|
|
|
2015-12-10 13:26:22 -05:00
|
|
|
index_remove(&playlist_db, playlist_names[plist],
|
|
|
|
track->tr_dbe.dbe_index);
|
|
|
|
if (playlist_cur == plist)
|
|
|
|
queue_remove_all(&playlist_q, track);
|
2015-12-11 08:21:21 -05:00
|
|
|
return true;
|
2013-12-22 17:53:58 -05:00
|
|
|
}
|
2013-12-22 20:53:06 -05:00
|
|
|
|
2015-12-10 13:26:22 -05:00
|
|
|
bool playlist_has(enum playlist_t plist, struct track *track)
|
2015-12-09 09:31:12 -05:00
|
|
|
{
|
2015-12-11 08:21:21 -05:00
|
|
|
if (!track || __playlist_is_dynamic(plist))
|
2015-12-09 09:31:12 -05:00
|
|
|
return false;
|
2015-12-10 13:26:22 -05:00
|
|
|
return index_has(&playlist_db, playlist_names[plist], track->tr_dbe.dbe_index);
|
2015-12-09 09:31:12 -05:00
|
|
|
}
|
|
|
|
|
2015-12-10 13:26:22 -05:00
|
|
|
void playlist_select(enum playlist_t plist)
|
2014-02-02 14:05:56 -05:00
|
|
|
{
|
2015-12-10 13:38:30 -05:00
|
|
|
queue_clear(&playlist_q);
|
|
|
|
queue_set_flag(&playlist_q, Q_ADD_FRONT);
|
2015-12-10 13:26:22 -05:00
|
|
|
if (__playlist_is_static(plist))
|
|
|
|
__playlist_fill_static(plist);
|
2015-12-10 13:38:30 -05:00
|
|
|
else
|
2015-12-10 13:26:22 -05:00
|
|
|
__playlist_fill_dynamic(plist);
|
2015-12-10 13:38:30 -05:00
|
|
|
queue_unset_flag(&playlist_q, Q_ADD_FRONT);
|
|
|
|
|
|
|
|
playlist_cur = plist;
|
|
|
|
queue_resort(&playlist_q);
|
2013-12-31 16:58:40 -05:00
|
|
|
}
|
|
|
|
|
2015-12-09 10:38:57 -05:00
|
|
|
struct queue *playlist_get_queue()
|
2013-12-31 16:58:40 -05:00
|
|
|
{
|
2014-05-18 21:09:46 -04:00
|
|
|
return &playlist_q;
|
2013-12-22 20:53:06 -05:00
|
|
|
}
|