Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Create ListBox-Sorted-StringList-model #1879

Open
wants to merge 11 commits into
base: main
Choose a base branch
from
144 changes: 144 additions & 0 deletions examples/ListBox-Sorted-StringList-model
Original file line number Diff line number Diff line change
@@ -0,0 +1,144 @@
// Variation on ListBox model example to show use of StringList and StringSorter.
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

That is not enough, you should look at other examples. You need a folder containing a main.rs file, then the folder should be added to the README.md and Cargo.toml; like other examples https://github.com/gtk-rs/gtk4-rs/blob/master/examples/Cargo.toml#L163

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

How do I create a folder? I'm afraid I'm not very familiar with actually doing stuff on github.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Feel free to ping me by DM on Matrix and can guide through that.

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'll do that tomorrow. Way past my bed time.

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think I have it figured out. I created a codespace where I can make the changes. Thanks for your help.

// Issues: list display doesn't update correctly on delete.

// TODO: add a dialog window to allow adding rows.
// use plain gtk::Window for this per current practice.

use gtk::{
glib::{self, clone},
prelude::*,
};

fn main() -> glib::ExitCode {
let application = gtk::Application::builder()
.application_id("com.github.gtk-rs.examples.listbox-sorted-StringList")
.build();

application.connect_activate(build_ui);

application.run()
}

fn build_ui(application: &gtk::Application) {
let window = gtk::ApplicationWindow::builder()
.default_width(320)
.default_height(480)
.application(application)
.title("Sorted StringList")
.build();

let vbox = gtk::Box::new(gtk::Orientation::Vertical, 5);

// Create a StringSorter with a property expression to sort
// StringObjects in a StringList. StringObject has a "string" property.
let expression = gtk::PropertyExpression::new(
gtk::StringObject::static_type(),
None::<gtk::Expression>,
"string");
let sorter = gtk::StringSorter::new(Some(expression));
sorter.set_ignore_case(true);

// Create our list store as a StringList and populate with some strings.
let model = gtk::StringList::new(&["zoo", "abba", "donkey", "sunrise", "river", "phoenix"]);

// Create a sort model and bind it to the ListStore and the sorter.
let sort_model = gtk::SortListModel::new(
Some(model.clone()),
Some(sorter.clone()));

// And then create the UI part, the listbox and bind the sort
// model to it. Whenever the UI needs to show a new row, e.g. because
// it was notified that the model changed, it will call the callback
// with the corresponding item from the model and will ask for a new
// gtk::ListBoxRow that should be displayed.
//
// The gtk::ListBoxRow can contain any possible widgets.
// Here we use an Inscription.

let listbox = gtk::ListBox::new();
listbox.bind_model(
Some(&sort_model),
clone!(
#[weak(rename_to = _panel)]
window,
#[upgrade_or_panic]
move |obj| {
let list_object = obj
.downcast_ref::<gtk::StringObject>()
.expect("The object should be of type `StringObject`.");
let row = gtk::Inscription::new(Some(&list_object.string()));
row.set_xalign(0.0);
row.upcast()
}
),
);

let scrolled_window = gtk::ScrolledWindow::builder()
.hscrollbar_policy(gtk::PolicyType::Never) // Disable horizontal scrolling
.min_content_height(480)
.min_content_width(360)
.build();

scrolled_window.set_child(Some(&listbox));

let hbox = gtk::Box::new(gtk::Orientation::Horizontal, 5);

// The add button opens a new dialog which is basically the same as the edit
// dialog, except that we don't have a corresponding item yet at that point
// and only create it once the Ok button in the dialog is clicked, and only
// then add it to the model. Once added to the model, it will immediately
// appear in the listbox UI
let add_button = gtk::Button::with_label("Add");
add_button.set_sensitive(false);

hbox.append(&add_button);

// Via the delete button we delete the item from the model that
// is at the index of the selected row. Also deleting from the
// model is immediately reflected in the listbox.
let delete_button = gtk::Button::with_label("Delete");
delete_button.connect_clicked(clone!(
#[weak]
model,
#[weak]
listbox,
move |_| {
let selected = listbox.selected_row();

if let Some(selected) = selected {
let selected = selected.child()
.expect("Listrow child should be a widget");
let selected = selected.downcast_ref::<gtk::Inscription>()
.expect("The object should be of type `Inscription`.");
if let Some(selected) = selected.text() {
let mut selected_index = None;
for ind in 0..model.n_items() {
if let Some(item) = model.item(ind) {
let item = item.downcast_ref::<gtk::StringObject>()
.expect("Object should be a stringobject");
if item.string() == selected {
selected_index = Some(ind);
break;
}
}
}
if let Some(index) = selected_index {
model.remove(index);
}
}
}
}
));
hbox.append(&delete_button);

vbox.append(&hbox);
vbox.append(&scrolled_window);

window.set_child(Some(&vbox));

for i in 0..10 {
model.append(&format!("Name {i}"));
}

window.present();
}
Loading