ocarina/lib/model.cpp

240 lines
4.4 KiB
C++

/*
* Copyright 2014 (c) Anna Schumaker.
*
* See the example at:
* https://git.gnome.org/browse/gtkmm-documentation/tree/examples/others/treemodelcustom
*/
#include <core/audio.h>
#include <lib/model.h>
#include <stdlib.h>
QueueModel::QueueModel(Queue *q)
: Glib::ObjectBase( typeid(QueueModel) ),
Glib::Object(), queue(q)
{
do {
stamp = rand();
} while (stamp == 0);
}
void QueueModel::increment_stamp()
{
do {
stamp++;
} while (stamp == 0);
}
void QueueModel::on_row_inserted(unsigned int row)
{
Gtk::TreePath path;
Gtk::TreeIter iter;
path.push_back(row);
increment_stamp();
row_inserted(path, iter);
}
void QueueModel::on_row_deleted(unsigned int row)
{
Gtk::TreePath path;
path.push_back(row);
increment_stamp();
row_deleted(path);
}
void QueueModel::on_row_changed(unsigned int row)
{
Gtk::TreePath path;
Gtk::TreeIter iter;
path.push_back(row);
increment_stamp();
row_changed(path, iter);
}
void QueueModel::on_path_selected(const Gtk::TreePath &path)
{
audio :: load_track(tags :: get_track(path_to_id(path)));
queue->track_selected(path[0]);
audio :: play();
}
unsigned int QueueModel :: iter_to_id(const Gtk::TreeIter &iter) const
{
return GPOINTER_TO_UINT(iter.gobj()->user_data);
}
unsigned int QueueModel::path_to_id(const Gtk::TreePath &path) const
{
return queue->operator[](path[0])->index();
}
/*
*
* Function overrides begin here
*
*/
Gtk::TreeModelFlags QueueModel::get_flags_vfunc() const
{
return Gtk::TREE_MODEL_LIST_ONLY;
}
int QueueModel::get_n_columns_vfunc() const
{
return 10;
}
GType QueueModel::get_column_type_vfunc(int index) const
{
switch (index) {
case 0:
case 5:
case 7:
return G_TYPE_UINT;
case 1:
case 2:
case 3:
case 4:
case 6:
case 8:
case 9:
return G_TYPE_STRING;
default:
return 0;
}
}
void QueueModel::get_value_uint(Track *track, int column,
Glib::ValueBase &value) const
{
Glib::Value<unsigned int> specific;
specific.init(Glib::Value<unsigned int>::value_type());
switch (column) {
case 0:
specific.set(track->track());
break;
case 5:
specific.set(track->album()->year());
break;
case 7:
specific.set(track->count());
}
value.init(Glib::Value<unsigned int>::value_type());
value = specific;
}
void QueueModel::get_value_str(Track *track, int column,
Glib::ValueBase &value) const
{
Glib::Value<std::string> specific;
specific.init(Glib::Value<std::string>::value_type());
switch (column) {
case 1:
specific.set(track->name());
break;
case 2:
specific.set(track->length_str());
break;
case 3:
specific.set(track->artist()->name());
break;
case 4:
specific.set(track->album()->name());
break;
case 6:
specific.set(track->genre()->name());
break;
case 8:
specific.set(track->date());
break;
case 9:
specific.set(Glib::Markup::escape_text(track->path()));
}
value.init(Glib::Value<std::string>::value_type());
value = specific;
}
void QueueModel::get_value_vfunc(const Gtk::TreeIter &iter, int column,
Glib::ValueBase &value) const
{
Track *track;
if (!check_iter_validity(iter))
return;
if (column > get_n_columns_vfunc())
return;
track = queue->operator[](iter_to_id(iter));
switch (column) {
case 0:
case 5:
case 7:
get_value_uint(track, column, value);
break;
default:
get_value_str(track, column, value);
}
}
bool QueueModel::iter_next_vfunc(const Gtk::TreeIter &iter,
Gtk::TreeIter &iter_next) const
{
if (!check_iter_validity(iter)) {
iter_next = Gtk::TreeIter();
return false;
}
return iter_nth_root_child_vfunc(iter_to_id(iter) + 1, iter_next);
}
int QueueModel::iter_n_root_children_vfunc() const
{
return queue->size();
}
bool QueueModel::iter_nth_root_child_vfunc(int n, Gtk::TreeIter &iter) const
{
iter = Gtk::TreeIter();
if (n >= (int)queue->size())
return false;
iter.set_stamp(stamp);
iter.gobj()->user_data = GUINT_TO_POINTER(n);
return true;
}
Gtk::TreeModel::Path QueueModel::get_path_vfunc(const Gtk::TreeIter &iter) const
{
Gtk::TreeModel::Path path;
if (check_iter_validity(iter))
path.push_back(iter_to_id(iter));
return path;
}
bool QueueModel::get_iter_vfunc(const Gtk::TreePath &path,
Gtk::TreeIter &iter) const
{
if (path.size() != 1) {
iter = Gtk::TreeIter();
return false;
}
return iter_nth_root_child_vfunc(path[0], iter);
}
bool QueueModel::check_iter_validity(const Gtk::TreeIter &iter) const
{
return stamp == iter.get_stamp();
}