1

I trying to do my first desktop app in Python and GTK3, but I quickly run into a problem. I want to display a TreeView with columns URL, Title, and delete icon, but I'm having problem in displaying and icon that can be clicked, and that deletes the row.

I found that question, but the solution didn't work for me.

Is there any way to do what I want? Or did I design it wrong?

Code:

    # List
    self.store = Gtk.ListStore(str, str, str)
    self.store.append(['https://www.youtube.com/watch?v=dQw4w9WgXcQ',  # URL
                       'Rick Astley - Never Gonna Give You Up',  # Title
                       'edit-delete'])  # Action icon
    tree = Gtk.TreeView(self.store)
    tree.set_size_request(600, 400)

    # Editable URL
    url = Gtk.CellRendererText()
    url.set_property("editable", True)
    url.connect("edited", self.text_edited)
    column_url = Gtk.TreeViewColumn("YouTube URL", url, text=0)
    column_url.set_min_width(300)
    tree.append_column(column_url)

    # Title
    title = Gtk.CellRendererText()
    column_title = Gtk.TreeViewColumn("Title", title, text=1)
    tree.append_column(column_title)

    # Action icon
    action_icon = Gtk.CellRendererPixbuf()
    # action_icon.connect("clicked", self.action_icon_clicked)
    column_action_icon = Gtk.TreeViewColumn("", action_icon, icon_name=2)
    tree.append_column(column_action_icon)

Thanks for help

pawel.ad
  • 691
  • 1
  • 6
  • 21

1 Answers1

1

The trick is to exploit the row activation in a Treeview to capture whether the button was clicked. As the row_activated tells you which column and row was clicked, such that we can delete the clicked row.

The default behaviour of a Treeview is double click to activate but it is possible to change this to single click using tree.set_activate_on_single_click(True). By now connecting to the listener to the signal like this tree.connect("row_activated", self.action_icon_clicked) we can use the function below to delete the clicked row.

def action_icon_clicked(self, treeview, path, column):
    # If the column clicked is the action column remove the clicked row
    if column is self.column_action_icon:

        # Get the iter that points to the clicked row
        iter = self.store.get_iter(path)

        # Remove it from the ListStore
        self.store.remove(iter)

So the complete code will become:

    # List
    self.store = Gtk.ListStore(str, str, str)
    self.store.append(['https://www.youtube.com/watch?v=dQw4w9WgXcQ',  # URL
                       'Rick Astley - Never Gonna Give You Up',  # Title
                       'edit-delete'])  # Action icon
    tree = Gtk.TreeView(self.store)
    tree.set_size_request(600, 400)

    # Editable URL
    url = Gtk.CellRendererText()
    url.set_property("editable", True)
    column_url = Gtk.TreeViewColumn("YouTube URL", url, text=0)
    column_url.set_min_width(300)
    tree.append_column(column_url)

    # Title
    title = Gtk.CellRendererText()
    column_title = Gtk.TreeViewColumn("Title", title, text=1)
    tree.append_column(column_title)

    # Action icon
    action_icon = Gtk.CellRendererPixbuf()
    self.column_action_icon = Gtk.TreeViewColumn("", action_icon, icon_name=2)
    tree.append_column(self.column_action_icon)

    # Make a click activate a row such that we get the row_activated signal when it is clicked
    tree.set_activate_on_single_click(True)

    # Connect a listener to the row_activated signal to check whether the correct column was clicked
    tree.connect("row_activated", self.action_icon_clicked)

def action_icon_clicked(self, treeview, path, column):
    # If the column clicked is the action column remove the clicked row
    if column is self.column_action_icon:

        # Get the iter that points to the clicked row
        iter = self.store.get_iter(path)

        # Remove it from the ListStore
        self.store.remove(iter)
B8vrede
  • 4,432
  • 3
  • 27
  • 40