ocarina/gui/tabs.cpp

229 lines
5.5 KiB
C++

/*
* Copyright 2014 (c) Anna Schumaker.
*/
#include <callback.h>
#include <deck.h>
#include <ocarina.h>
#include <playqueue.h>
#include <map>
#include <sstream>
#include <string>
class OcarinaTab;
static std::map<Playqueue *, OcarinaTab *> tab_map;
static class QueueColumns : public Gtk::TreeModelColumnRecord {
public:
QueueColumns()
{ add(q_col_track); add(q_col_title); add(q_col_length);
add(q_col_artist); add(q_col_album); add(q_col_year);
add(q_col_genre); add(q_col_count); add(q_col_played); }
Gtk::TreeModelColumn<unsigned int> q_col_track;
Gtk::TreeModelColumn<std::string> q_col_title;
Gtk::TreeModelColumn<std::string> q_col_length;
Gtk::TreeModelColumn<std::string> q_col_artist;
Gtk::TreeModelColumn<std::string> q_col_album;
Gtk::TreeModelColumn<unsigned int> q_col_year;
Gtk::TreeModelColumn<std::string> q_col_genre;
Gtk::TreeModelColumn<unsigned int> q_col_count;
Gtk::TreeModelColumn<std::string> q_col_played;
} queue_cols;
static unsigned int q_col_width[] = { 20, 300, 60, 125, 125, 1, 125, 60, 1 };
/*
* Tab class definition
*/
class OcarinaTab {
private:
Glib::RefPtr<PlayqueueModel> model;
Gtk::Notebook *notebook;
/* Tab widgets */
Gtk::VBox tab_box;
Gtk::Label tab_name;
Gtk::Label tab_size;
/* Page widgets */
Gtk::VBox page_box;
Gtk::HBox page_toolbar;
Gtk::SearchEntry page_entry;
Gtk::ToggleButton page_random;
//Gtk::ToggleButton page_repeat;
Gtk::ScrolledWindow page_scroll;
Gtk::TreeView page_view;
void setup_columns();
void set_tab_size();
public:
OcarinaTab(const std::string &, Playqueue *);
~OcarinaTab();
bool is_current_tab();
bool is_current_tab(unsigned int);
void on_row_inserted(unsigned int);
void on_row_deleted(unsigned int);
void on_runtime_changed();
};
OcarinaTab::OcarinaTab(const std::string &text, Playqueue *pq)
: tab_name("<big>" + text + "</big>", 1, 0.5),
tab_size("0", 1, 0.5)
{
get_builder()->get_widget("o_notebook", notebook);
model = Glib::RefPtr<PlayqueueModel>(new PlayqueueModel(pq));
/* Make tab label */
tab_name.set_use_markup();
tab_box.pack_start(tab_name);
tab_box.pack_start(tab_size);
tab_box.show_all();
set_tab_size();
/* Make page content */
page_random.set_image_from_icon_name("media-playlist-shuffle");
page_random.set_active(pq->get_flags() & PQ_RANDOM);
//page_repeat.set_image_from_icon_name("media-playlist-repeat");
page_toolbar.set_spacing(5);
page_view.append_column("#", queue_cols.q_col_track);
page_view.append_column("Title", queue_cols.q_col_title);
page_view.append_column("Length", queue_cols.q_col_length);
page_view.append_column("Artist", queue_cols.q_col_artist);
page_view.append_column("Album", queue_cols.q_col_album);
page_view.append_column("Year", queue_cols.q_col_year);
page_view.append_column("Genre", queue_cols.q_col_genre);
page_view.append_column("Count", queue_cols.q_col_count);
page_view.append_column("Played", queue_cols.q_col_played);
page_view.set_model(model);
page_scroll.add(page_view);
setup_columns();
page_toolbar.pack_start(page_entry);
page_toolbar.pack_start(page_random, false, false);
//page_toolbar.pack_start(page_repeat, false, false);
page_box.pack_start(page_toolbar, false, false);
page_box.pack_start(page_scroll);
page_box.show_all();
/* Add to notebook */
notebook->prepend_page(page_box, tab_box);
tab_map[pq] = this;
};
OcarinaTab::~OcarinaTab()
{
notebook->remove_page(page_box);
tab_map.erase(model->queue);
}
void OcarinaTab::setup_columns()
{
std::vector<Gtk::TreeViewColumn *> columns = page_view.get_columns();
for (unsigned int i = 0; i < columns.size(); i++) {
columns[i]->set_resizable();
columns[i]->set_fixed_width(q_col_width[i]);
columns[i]->set_sizing(Gtk::TREE_VIEW_COLUMN_FIXED);
}
}
bool OcarinaTab::is_current_tab()
{
return notebook->page_num(page_box) == notebook->get_current_page();
}
bool OcarinaTab::is_current_tab(unsigned int tab)
{
return notebook->page_num(page_box) == (int)tab;
}
void OcarinaTab::set_tab_size()
{
std::stringstream ss;
ss << model->queue->size();
tab_size.set_text(ss.str());
}
void OcarinaTab::on_row_inserted(unsigned int row)
{
model->on_row_inserted(row);
set_tab_size();
if (is_current_tab())
on_runtime_changed();
}
void OcarinaTab::on_row_deleted(unsigned int row)
{
model->on_row_deleted(row);
set_tab_size();
if (is_current_tab())
on_runtime_changed();
}
void OcarinaTab::on_runtime_changed()
{
Gtk::Label *label;
get_builder()->get_widget("o_queue_time", label);
label->set_text(model->queue->get_length_str());
}
/*
* Do stuff with tabs
*/
static void on_track_added(Playqueue *pq, unsigned int row)
{
tab_map[pq]->on_row_inserted(row);
}
static void on_track_deleted(Playqueue *pq, unsigned int row)
{
tab_map[pq]->on_row_deleted(row);
}
static void on_switch_page(Gtk::Widget *page, unsigned int num)
{
Gtk::Label *label;
std::map<Playqueue *, OcarinaTab *>::iterator it;
get_builder()->get_widget("o_queue_time", label);
for (it = tab_map.begin(); it != tab_map.end(); it++) {
if (it->second->is_current_tab(num)) {
it->second->on_runtime_changed();
label->show();
return;
}
}
label->hide();
}
void init_tabs()
{
Gtk::Notebook *notebook;
get_builder()->get_widget("o_notebook", notebook);
new OcarinaTab("Collection", deck::get_library_pq());
get_callbacks()->on_queue_track_add = on_track_added;
get_callbacks()->on_queue_track_del = on_track_deleted;
notebook->signal_switch_page().connect(sigc::ptr_fun(on_switch_page));
}
void cleanup_tabs()
{
std::map<Playqueue *, OcarinaTab *>::iterator it;
for (it = tab_map.begin(); it != tab_map.end(); it++)
delete it->second;
}