mirror of
https://gitlab.gnome.org/GNOME/gitg
synced 2024-10-14 11:52:29 +00:00
530 lines
11 KiB
Vala
530 lines
11 KiB
Vala
/*
|
|
* This file is part of gitg
|
|
*
|
|
* Copyright (C) 2012-2016 - Ignacio Casal Quinteiro
|
|
*
|
|
* gitg is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* gitg is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with gitg. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
namespace Gitg
|
|
{
|
|
public enum SelectionMode
|
|
{
|
|
NORMAL,
|
|
SELECTION
|
|
}
|
|
|
|
public class RepositoryListBox : Gtk.ListBox
|
|
{
|
|
private string? d_filter_text;
|
|
|
|
public signal void repository_activated(Repository repository);
|
|
public signal void show_error(string primary_message, string secondary_message);
|
|
|
|
[GtkTemplate (ui = "/org/gnome/gitg/ui/gitg-repository-list-box-row.ui")]
|
|
public class Row : Gtk.ListBoxRow
|
|
{
|
|
private Repository? d_repository;
|
|
private DateTime d_time;
|
|
private bool d_loading;
|
|
[GtkChild]
|
|
private ProgressBin d_progress_bin;
|
|
[GtkChild]
|
|
private Gtk.Label d_repository_label;
|
|
[GtkChild]
|
|
private Gtk.Label d_description_label;
|
|
[GtkChild]
|
|
private Gtk.Label d_branch_label;
|
|
[GtkChild]
|
|
private Gtk.Spinner d_spinner;
|
|
[GtkChild]
|
|
private Gtk.CheckButton d_remove_check_button;
|
|
[GtkChild]
|
|
private Gtk.Revealer d_remove_revealer;
|
|
[GtkChild]
|
|
private Gtk.Box d_languages_box;
|
|
|
|
public signal void request_remove();
|
|
|
|
private SelectionMode d_mode;
|
|
private string? d_dirname;
|
|
private string? d_branch_name;
|
|
|
|
public SelectionMode mode
|
|
{
|
|
get { return d_mode; }
|
|
|
|
set
|
|
{
|
|
if (d_mode != value)
|
|
{
|
|
d_mode = value;
|
|
|
|
d_remove_revealer.reveal_child = (d_mode == SelectionMode.SELECTION);
|
|
|
|
d_remove_check_button.active = false;
|
|
}
|
|
}
|
|
}
|
|
|
|
public new bool selected
|
|
{
|
|
get; set;
|
|
}
|
|
|
|
construct
|
|
{
|
|
d_remove_check_button.bind_property("active",
|
|
this,
|
|
"selected",
|
|
BindingFlags.BIDIRECTIONAL |
|
|
BindingFlags.SYNC_CREATE);
|
|
}
|
|
|
|
public Repository? repository
|
|
{
|
|
get { return d_repository; }
|
|
set
|
|
{
|
|
d_repository = value;
|
|
update_repository_data();
|
|
}
|
|
}
|
|
|
|
public bool can_remove
|
|
{
|
|
get { return d_remove_check_button.sensitive; }
|
|
set { d_remove_check_button.sensitive = value; }
|
|
}
|
|
|
|
public DateTime time
|
|
{
|
|
get { return d_time; }
|
|
set { d_time = value; }
|
|
default = new DateTime.now_local();
|
|
}
|
|
|
|
public double fraction
|
|
{
|
|
set { d_progress_bin.fraction = value; }
|
|
}
|
|
|
|
public string? repository_name
|
|
{
|
|
get { return d_repository_label.get_text(); }
|
|
set { d_repository_label.label = value; }
|
|
}
|
|
|
|
public string? dirname
|
|
{
|
|
get { return d_dirname; }
|
|
set
|
|
{
|
|
d_dirname = value;
|
|
update_branch_label();
|
|
}
|
|
}
|
|
|
|
public string? branch_name
|
|
{
|
|
get { return d_branch_name; }
|
|
set
|
|
{
|
|
d_branch_name = value;
|
|
update_branch_label();
|
|
}
|
|
}
|
|
|
|
private void update_branch_label()
|
|
{
|
|
if (d_branch_name == null || d_branch_name == "")
|
|
{
|
|
// Translators: this is used to construct: "at <directory>", to indicate where the repository is at.
|
|
d_branch_label.label = _("at %s").printf(d_dirname);
|
|
}
|
|
else if (d_dirname == null || d_dirname == "")
|
|
{
|
|
d_branch_label.label = d_branch_name;
|
|
}
|
|
else
|
|
{
|
|
// Translators: this is used to construct: "<branch-name> at <directory>"
|
|
d_branch_label.label = _("%s at %s").printf(d_branch_name, d_dirname);
|
|
}
|
|
}
|
|
|
|
private void update_repository_data()
|
|
{
|
|
string head_name = "";
|
|
string head_description = "";
|
|
|
|
if (d_repository != null)
|
|
{
|
|
try
|
|
{
|
|
var head = d_repository.get_head();
|
|
head_name = head.parsed_name.shortname;
|
|
|
|
var commit = (Ggit.Commit)head.lookup();
|
|
var tree = commit.get_tree();
|
|
|
|
Ggit.OId? entry_id = null;
|
|
|
|
for (var i = 0; i < tree.size(); i++)
|
|
{
|
|
var entry = tree.get(i);
|
|
var name = entry.get_name();
|
|
|
|
if (name != null && name.has_suffix(".doap"))
|
|
{
|
|
entry_id = entry.get_id();
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (entry_id != null)
|
|
{
|
|
var blob = d_repository.lookup<Ggit.Blob>(entry_id);
|
|
|
|
unowned uint8[] content = blob.get_raw_content();
|
|
var doap = new Ide.Doap();
|
|
doap.load_from_data((string)content, -1);
|
|
|
|
head_description = doap.get_shortdesc();
|
|
|
|
foreach (var lang in doap.get_languages())
|
|
{
|
|
var frame = new Gtk.Frame(null);
|
|
frame.shadow_type = Gtk.ShadowType.NONE;
|
|
frame.get_style_context().add_class("language-frame");
|
|
frame.show();
|
|
|
|
var label = new Gtk.Label(lang);
|
|
var attr_list = new Pango.AttrList();
|
|
attr_list.insert(Pango.attr_scale_new(Pango.Scale.SMALL));
|
|
label.set_attributes(attr_list);
|
|
label.show();
|
|
|
|
frame.add(label);
|
|
d_languages_box.add(frame);
|
|
}
|
|
}
|
|
} catch {}
|
|
}
|
|
|
|
repository_name = d_repository != null ? d_repository.name : "";
|
|
|
|
d_description_label.label = head_description;
|
|
d_description_label.visible = head_description != "";
|
|
|
|
branch_name = head_name;
|
|
}
|
|
|
|
public bool loading
|
|
{
|
|
get { return d_loading; }
|
|
set
|
|
{
|
|
d_loading = value;
|
|
|
|
if (!d_loading)
|
|
{
|
|
d_spinner.stop();
|
|
d_spinner.hide();
|
|
d_progress_bin.fraction = 0;
|
|
}
|
|
else
|
|
{
|
|
d_spinner.show();
|
|
d_spinner.start();
|
|
}
|
|
}
|
|
}
|
|
|
|
public Row(Repository? repository, string dirname)
|
|
{
|
|
Object(repository: repository, dirname: dirname);
|
|
}
|
|
}
|
|
|
|
public SelectionMode mode { get; set; }
|
|
|
|
protected override bool button_press_event(Gdk.EventButton event)
|
|
{
|
|
Gdk.Event *ev = (Gdk.Event *)event;
|
|
|
|
if (ev->triggers_context_menu() && mode == SelectionMode.NORMAL)
|
|
{
|
|
mode = SelectionMode.SELECTION;
|
|
|
|
var row = get_row_at_y((int)event.y) as Row;
|
|
|
|
if (row != null)
|
|
{
|
|
row.selected = true;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
return base.button_press_event(event);
|
|
}
|
|
|
|
protected override void row_activated(Gtk.ListBoxRow row)
|
|
{
|
|
var r = (Row)row;
|
|
|
|
if (mode == SelectionMode.SELECTION)
|
|
{
|
|
r.selected = !r.selected;
|
|
return;
|
|
}
|
|
|
|
if (r.repository != null)
|
|
{
|
|
repository_activated(r.repository);
|
|
}
|
|
}
|
|
|
|
construct
|
|
{
|
|
set_header_func(update_header);
|
|
set_filter_func(filter);
|
|
set_sort_func(compare_widgets);
|
|
show();
|
|
|
|
set_selection_mode(Gtk.SelectionMode.NONE);
|
|
}
|
|
|
|
private void update_header(Gtk.ListBoxRow row, Gtk.ListBoxRow? before)
|
|
{
|
|
row.set_header(before != null ? new Gtk.Separator(Gtk.Orientation.HORIZONTAL) : null);
|
|
}
|
|
|
|
private bool filter(Gtk.ListBoxRow row)
|
|
{
|
|
return d_filter_text != null ? ((Row)row).repository_name.contains(d_filter_text) : true;
|
|
}
|
|
|
|
private int compare_widgets(Gtk.ListBoxRow a, Gtk.ListBoxRow b)
|
|
{
|
|
return - ((Row)a).time.compare(((Row)b).time);
|
|
}
|
|
|
|
public void populate_recent()
|
|
{
|
|
add_recent_info();
|
|
}
|
|
|
|
private void add_recent_info()
|
|
{
|
|
var recent_manager = Gtk.RecentManager.get_default();
|
|
var reversed_items = recent_manager.get_items();
|
|
reversed_items.reverse();
|
|
|
|
foreach (var item in reversed_items)
|
|
{
|
|
if (item.has_group("gitg"))
|
|
{
|
|
File repo_file = File.new_for_uri(item.get_uri());
|
|
Repository repo;
|
|
|
|
try
|
|
{
|
|
repo = new Repository(repo_file, null);
|
|
}
|
|
catch
|
|
{
|
|
try
|
|
{
|
|
recent_manager.remove_item(item.get_uri());
|
|
}
|
|
catch {}
|
|
continue;
|
|
}
|
|
|
|
add_repository(repo);
|
|
}
|
|
}
|
|
}
|
|
|
|
private Row get_row_for_repository(Repository repository)
|
|
{
|
|
Row? row = null;
|
|
|
|
foreach (var child in get_children())
|
|
{
|
|
var d = (Row)child;
|
|
|
|
if (d.repository.get_location().equal(repository.get_location()))
|
|
{
|
|
row = d;
|
|
break;
|
|
}
|
|
}
|
|
|
|
return row;
|
|
}
|
|
|
|
private void add_repository_to_recent_manager(string uri)
|
|
{
|
|
var recent_manager = Gtk.RecentManager.get_default();
|
|
var item = Gtk.RecentData();
|
|
|
|
item.app_name = Environment.get_application_name();
|
|
item.mime_type = "inode/directory";
|
|
item.app_exec = string.join(" ", Environment.get_prgname(), "%f");
|
|
item.groups = { "gitg", null };
|
|
|
|
recent_manager.add_full(uri, item);
|
|
}
|
|
|
|
public void end_cloning(Row row, Repository? repository)
|
|
{
|
|
if (repository != null)
|
|
{
|
|
File? workdir = repository.get_workdir();
|
|
File? repo_file = repository.get_location();
|
|
|
|
var uri = (workdir != null) ? workdir.get_uri() : repo_file.get_uri();
|
|
add_repository_to_recent_manager(uri);
|
|
|
|
row.repository = repository;
|
|
row.loading = false;
|
|
}
|
|
else
|
|
{
|
|
remove(row);
|
|
}
|
|
}
|
|
|
|
public Row? begin_cloning(File location)
|
|
{
|
|
var row = new Row(null, Utils.replace_home_dir_with_tilde(location.get_parent()));
|
|
row.repository_name = location.get_basename();
|
|
row.branch_name = _("Cloning…");
|
|
|
|
row.loading = true;
|
|
row.show();
|
|
|
|
add(row);
|
|
return row;
|
|
}
|
|
|
|
public Row? add_repository(Repository repository)
|
|
{
|
|
Row? row = get_row_for_repository(repository);
|
|
|
|
var f = repository.workdir != null ? repository.workdir : repository.location;
|
|
|
|
if (row == null)
|
|
{
|
|
var dirname = Utils.replace_home_dir_with_tilde((repository.workdir != null ? repository.workdir : repository.location).get_parent());
|
|
row = new Row(repository, dirname);
|
|
row.show();
|
|
|
|
if (f != null)
|
|
{
|
|
bind_property("mode", row, "mode");
|
|
}
|
|
|
|
if (f != null)
|
|
{
|
|
row.notify["selected"].connect(() => {
|
|
notify_property("has-selection");
|
|
});
|
|
|
|
row.request_remove.connect(() => {
|
|
try
|
|
{
|
|
var recent_manager = Gtk.RecentManager.get_default();
|
|
recent_manager.remove_item(f.get_uri());
|
|
} catch {}
|
|
|
|
remove(row);
|
|
});
|
|
|
|
row.can_remove = true;
|
|
}
|
|
else
|
|
{
|
|
row.can_remove = false;
|
|
}
|
|
|
|
add(row);
|
|
}
|
|
else
|
|
{
|
|
// to get the item sorted to the beginning of the list
|
|
row.time = new DateTime.now_local();
|
|
invalidate_sort();
|
|
}
|
|
|
|
if (f != null)
|
|
{
|
|
add_repository_to_recent_manager(f.get_uri());
|
|
}
|
|
|
|
return row;
|
|
}
|
|
|
|
public Row[] selection
|
|
{
|
|
owned get
|
|
{
|
|
var ret = new Row[0];
|
|
|
|
foreach (var row in get_children())
|
|
{
|
|
var r = (Row)row;
|
|
|
|
if (r.selected)
|
|
{
|
|
ret += r;
|
|
}
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
}
|
|
|
|
public bool has_selection
|
|
{
|
|
get
|
|
{
|
|
foreach (var row in get_children())
|
|
{
|
|
var r = (Row)row;
|
|
|
|
if (r.selected)
|
|
{
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
}
|
|
|
|
public void filter_text(string? text)
|
|
{
|
|
d_filter_text = text;
|
|
|
|
invalidate_filter();
|
|
}
|
|
}
|
|
}
|
|
|
|
// ex:ts=4 noet
|