+static void ql_remove_sensitive_callback(GtkTreeModel *model,
+ GtkTreePath attribute((unused)) *path,
+ GtkTreeIter *iter,
+ gpointer data) {
+ struct queue_entry *q = ql_iter_to_q(model, iter);
+ const int removable = (q != playing_track
+ && right_removable(last_rights, config->username, q));
+ int *const counts = data;
+ ++counts[removable];
+}
+
+int ql_remove_sensitive(void *extra) {
+ struct queuelike *ql = extra;
+ int counts[2] = { 0, 0 };
+ gtk_tree_selection_selected_foreach(ql->selection,
+ ql_remove_sensitive_callback,
+ counts);
+ /* Remove will work if we have at least some removable tracks selected, and
+ * no unremovable ones */
+ return counts[1] > 0 && counts[0] == 0;
+}
+
+static void ql_remove_completed(void attribute((unused)) *v,
+ const char *err) {
+ if(err)
+ popup_protocol_error(0, err);
+}
+
+static void ql_remove_activate_callback(GtkTreeModel *model,
+ GtkTreePath attribute((unused)) *path,
+ GtkTreeIter *iter,
+ gpointer attribute((unused)) data) {
+ struct queue_entry *q = ql_iter_to_q(model, iter);
+
+ if(q != playing_track)
+ disorder_eclient_remove(client, ql_remove_completed, q->id, q);