mirror of https://github.com/bsnes-emu/bsnes.git
212 lines
7.7 KiB
C++
Executable File
212 lines
7.7 KiB
C++
Executable File
static void ListView_activate(ListView *self) {
|
|
if(self->onActivate) self->onActivate();
|
|
}
|
|
|
|
static void ListView_change(ListView *self) {
|
|
if(self->state.selected == false || self->state.selection != self->selection()) {
|
|
self->state.selected = true;
|
|
self->state.selection = self->selection();
|
|
if(self->onChange) self->onChange();
|
|
}
|
|
}
|
|
|
|
static void ListView_tick(GtkCellRendererToggle *cell, gchar *path_string, ListView *self) {
|
|
unsigned row = decimal(path_string);
|
|
self->setChecked(row, !self->checked(row));
|
|
if(self->onTick) self->onTick(row);
|
|
}
|
|
|
|
void pListView::append(const lstring &text) {
|
|
GtkTreeIter iter;
|
|
gtk_list_store_append(store, &iter);
|
|
foreach(item, text, n) gtk_list_store_set(store, &iter, 1 + n, (const char*)item, -1);
|
|
}
|
|
|
|
void pListView::autoSizeColumns() {
|
|
gtk_tree_view_columns_autosize(GTK_TREE_VIEW(subWidget));
|
|
}
|
|
|
|
bool pListView::checked(unsigned row) {
|
|
GtkTreeModel *model = gtk_tree_view_get_model(GTK_TREE_VIEW(subWidget));
|
|
GtkTreePath *path = gtk_tree_path_new_from_string(string(row));
|
|
GtkTreeIter iter;
|
|
bool state;
|
|
gtk_tree_model_get_iter(model, &iter, path);
|
|
gtk_tree_model_get(model, &iter, 0, &state, -1);
|
|
gtk_tree_path_free(path);
|
|
return state;
|
|
}
|
|
|
|
void pListView::modify(unsigned row, const lstring &text) {
|
|
GtkTreeModel *model = gtk_tree_view_get_model(GTK_TREE_VIEW(subWidget));
|
|
GtkTreeIter iter;
|
|
for(unsigned i = 0; i <= row; i++) {
|
|
if(i == 0) gtk_tree_model_get_iter_first(model, &iter);
|
|
else gtk_tree_model_iter_next(model, &iter);
|
|
}
|
|
foreach(item, text, n) gtk_list_store_set(store, &iter, 1 + n, (const char*)item, -1);
|
|
}
|
|
|
|
void pListView::reset() {
|
|
listView.state.selected = false;
|
|
listView.state.selection = 0;
|
|
gtk_list_store_clear(GTK_LIST_STORE(store));
|
|
gtk_tree_view_set_model(GTK_TREE_VIEW(subWidget), GTK_TREE_MODEL(store));
|
|
//reset gtk_scrolled_window scrollbar position to 0,0 (top-left), as ListView is now empty
|
|
gtk_scrolled_window_set_hadjustment(GTK_SCROLLED_WINDOW(gtkWidget), 0);
|
|
gtk_scrolled_window_set_vadjustment(GTK_SCROLLED_WINDOW(gtkWidget), 0);
|
|
}
|
|
|
|
bool pListView::selected() {
|
|
GtkTreeSelection *selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(subWidget));
|
|
GtkTreeModel *model = gtk_tree_view_get_model(GTK_TREE_VIEW(subWidget));
|
|
GtkTreeIter iter;
|
|
if(gtk_tree_model_get_iter_first(model, &iter) == false) return false;
|
|
if(gtk_tree_selection_iter_is_selected(selection, &iter) == true) return true;
|
|
for(unsigned n = 1;; n++) {
|
|
if(gtk_tree_model_iter_next(model, &iter) == false) return false;
|
|
if(gtk_tree_selection_iter_is_selected(selection, &iter) == true) return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
unsigned pListView::selection() {
|
|
if(selected() == false) return listView.state.selection;
|
|
|
|
GtkTreeSelection *selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(subWidget));
|
|
GtkTreeModel *model = gtk_tree_view_get_model(GTK_TREE_VIEW(subWidget));
|
|
GtkTreeIter iter;
|
|
if(gtk_tree_model_get_iter_first(model, &iter) == false) return 0;
|
|
if(gtk_tree_selection_iter_is_selected(selection, &iter) == true) return 0;
|
|
for(unsigned n = 1;; n++) {
|
|
if(gtk_tree_model_iter_next(model, &iter) == false) return 0;
|
|
if(gtk_tree_selection_iter_is_selected(selection, &iter) == true) return n;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
void pListView::setCheckable(bool checkable) {
|
|
gtk_tree_view_column_set_visible(column[0].column, checkable);
|
|
}
|
|
|
|
void pListView::setChecked(unsigned row, bool checked) {
|
|
GtkTreeModel *model = gtk_tree_view_get_model(GTK_TREE_VIEW(subWidget));
|
|
GtkTreePath *path = gtk_tree_path_new_from_string(string(row));
|
|
GtkTreeIter iter;
|
|
gtk_tree_model_get_iter(model, &iter, path);
|
|
gtk_list_store_set(GTK_LIST_STORE(model), &iter, 0, checked, -1);
|
|
gtk_tree_path_free(path);
|
|
}
|
|
|
|
void pListView::setHeaderText(const lstring &text) {
|
|
create();
|
|
}
|
|
|
|
void pListView::setHeaderVisible(bool visible) {
|
|
gtk_tree_view_set_headers_visible(GTK_TREE_VIEW(subWidget), visible);
|
|
}
|
|
|
|
void pListView::setSelected(bool selected) {
|
|
if(selected == false) {
|
|
GtkTreeSelection *selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(subWidget));
|
|
gtk_tree_selection_unselect_all(selection);
|
|
} else {
|
|
setSelection(listView.state.selection);
|
|
}
|
|
}
|
|
|
|
void pListView::setSelection(unsigned row) {
|
|
signed current = -1;
|
|
if(selected()) current = selection();
|
|
GtkTreeSelection *selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(subWidget));
|
|
GtkTreeModel *model = gtk_tree_view_get_model(GTK_TREE_VIEW(subWidget));
|
|
gtk_tree_selection_unselect_all(selection);
|
|
|
|
GtkTreeIter iter;
|
|
if(gtk_tree_model_get_iter_first(model, &iter) == false) return;
|
|
if(row == 0) {
|
|
gtk_tree_selection_select_iter(selection, &iter);
|
|
return;
|
|
}
|
|
for(unsigned n = 1;; n++) {
|
|
if(gtk_tree_model_iter_next(model, &iter) == false) return;
|
|
if(row == n) {
|
|
gtk_tree_selection_select_iter(selection, &iter);
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
void pListView::constructor() {
|
|
gtkWidget = 0;
|
|
subWidget = 0;
|
|
create();
|
|
}
|
|
|
|
void pListView::create() {
|
|
if(subWidget) gtk_widget_destroy(subWidget);
|
|
if(gtkWidget) gtk_widget_destroy(gtkWidget);
|
|
|
|
gtkWidget = gtk_scrolled_window_new(0, 0);
|
|
gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(gtkWidget), GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC);
|
|
gtk_scrolled_window_set_shadow_type(GTK_SCROLLED_WINDOW(gtkWidget), GTK_SHADOW_ETCHED_IN);
|
|
|
|
lstring headerText;
|
|
headerText.append(""); //checkbox column
|
|
foreach(headerItem, listView.state.headerText) headerText.append(headerItem);
|
|
if(headerText.size() == 1) headerText.append("");
|
|
|
|
GType *v = (GType*)malloc(headerText.size() * sizeof(GType));
|
|
foreach(header, headerText, n) v[n] = (n == 0 ? G_TYPE_BOOLEAN : G_TYPE_STRING);
|
|
store = gtk_list_store_newv(headerText.size(), v);
|
|
free(v);
|
|
|
|
subWidget = gtk_tree_view_new_with_model(GTK_TREE_MODEL(store));
|
|
gtk_container_add(GTK_CONTAINER(gtkWidget), subWidget);
|
|
g_object_unref(G_OBJECT(store));
|
|
|
|
foreach(header, headerText, n) {
|
|
if(n == 0) {
|
|
column[n].renderer = gtk_cell_renderer_toggle_new();
|
|
column[n].column = gtk_tree_view_column_new_with_attributes("", column[n].renderer, "active", n, (void*)0);
|
|
gtk_tree_view_column_set_resizable(column[n].column, false);
|
|
gtk_tree_view_column_set_visible(column[n].column, false);
|
|
g_signal_connect(column[n].renderer, "toggled", G_CALLBACK(ListView_tick), (gpointer)&listView);
|
|
} else {
|
|
column[n].renderer = gtk_cell_renderer_text_new();
|
|
column[n].column = gtk_tree_view_column_new_with_attributes("", column[n].renderer, "text", n, (void*)0);
|
|
gtk_tree_view_column_set_resizable(column[n].column, true);
|
|
}
|
|
column[n].label = gtk_label_new(header);
|
|
gtk_tree_view_column_set_widget(GTK_TREE_VIEW_COLUMN(column[n].column), column[n].label);
|
|
gtk_tree_view_append_column(GTK_TREE_VIEW(subWidget), column[n].column);
|
|
gtk_widget_show(column[n].label);
|
|
}
|
|
|
|
gtk_tree_view_set_rules_hint(GTK_TREE_VIEW(subWidget), headerText.size() >= 3); //two or more columns + checkbox column
|
|
gtk_tree_view_set_search_column(GTK_TREE_VIEW(subWidget), 1);
|
|
|
|
g_signal_connect_swapped(G_OBJECT(subWidget), "cursor-changed", G_CALLBACK(ListView_change), (gpointer)&listView);
|
|
g_signal_connect_swapped(G_OBJECT(subWidget), "row-activated", G_CALLBACK(ListView_activate), (gpointer)&listView);
|
|
|
|
setHeaderVisible(listView.state.headerVisible);
|
|
setCheckable(listView.state.checkable);
|
|
foreach(text, listView.state.text) append(text);
|
|
foreach(checked, listView.state.checked, n) setChecked(n, checked);
|
|
if(listView.state.selected) setSelection(listView.state.selection);
|
|
autoSizeColumns();
|
|
|
|
gtk_widget_show(subWidget);
|
|
}
|
|
|
|
void pListView::setFocused() {
|
|
gtk_widget_grab_focus(subWidget);
|
|
}
|
|
|
|
void pListView::setFont(Font &font) {
|
|
pWidget::setFont(font);
|
|
for(unsigned n = 0; n < 1 + listView.state.headerText.size(); n++) {
|
|
gtk_widget_modify_font(column[n].label, font.p.gtkFont);
|
|
}
|
|
}
|