2020-09-21 20:28:43 +02:00
|
|
|
extends Control
|
2020-09-01 17:24:21 +02:00
|
|
|
|
2020-09-29 14:26:20 +02:00
|
|
|
onready var index = $Rows/RepoVis/Index
|
|
|
|
onready var nodes = $Rows/RepoVis/Nodes
|
|
|
|
onready var file_browser = $Rows/FileBrowser
|
|
|
|
onready var label_node = $Rows/RepoVis/Label
|
2020-09-29 17:20:24 +02:00
|
|
|
onready var path_node = $Rows/RepoVis/Path
|
2020-09-29 14:26:20 +02:00
|
|
|
onready var simplify_checkbox = $Rows/RepoVis/SimplifyCheckbox
|
2020-09-22 13:15:36 +02:00
|
|
|
|
2020-09-03 18:10:09 +02:00
|
|
|
export var label: String setget set_label
|
2020-09-01 17:24:21 +02:00
|
|
|
export var path: String setget set_path, get_path
|
2020-09-22 13:15:36 +02:00
|
|
|
export var file_browser_active = true setget set_file_browser_active
|
2020-09-23 12:05:43 +02:00
|
|
|
export var simplified_view = false setget set_simplified_view
|
2020-09-29 17:20:24 +02:00
|
|
|
export var editable_path = false setget set_editable_path
|
2020-09-03 17:50:03 +02:00
|
|
|
|
2020-09-01 17:24:21 +02:00
|
|
|
var node = preload("res://node.tscn")
|
|
|
|
|
2020-09-08 16:00:18 +02:00
|
|
|
var shell = Shell.new()
|
|
|
|
var objects = {}
|
2020-09-15 18:41:06 +02:00
|
|
|
var mouse_inside = false
|
2020-09-08 16:00:18 +02:00
|
|
|
|
2020-09-30 22:41:28 +02:00
|
|
|
# We use this for a heuristic of when to hide trees and blobs.
|
|
|
|
var _commit_count = 0
|
|
|
|
|
2020-09-01 17:24:21 +02:00
|
|
|
func _ready():
|
2020-09-22 13:15:36 +02:00
|
|
|
file_browser.shell = shell
|
|
|
|
|
|
|
|
# Trigger these again because nodes were not ready before.
|
|
|
|
set_label(label)
|
|
|
|
set_file_browser_active(file_browser_active)
|
2020-09-23 12:05:43 +02:00
|
|
|
set_simplified_view(simplified_view)
|
2020-09-29 17:20:24 +02:00
|
|
|
set_editable_path(editable_path)
|
2020-09-30 21:36:11 +02:00
|
|
|
set_path(path)
|
|
|
|
|
|
|
|
update_everything()
|
2020-10-01 13:11:29 +02:00
|
|
|
update_node_positions()
|
2020-09-01 17:24:21 +02:00
|
|
|
|
2020-09-08 20:26:14 +02:00
|
|
|
func _process(_delta):
|
2020-09-22 21:56:22 +02:00
|
|
|
nodes.rect_pivot_offset = nodes.rect_size / 2
|
2020-09-01 18:26:43 +02:00
|
|
|
if path:
|
2020-09-01 19:32:33 +02:00
|
|
|
apply_forces()
|
2020-09-01 19:20:51 +02:00
|
|
|
|
2020-09-21 18:53:36 +02:00
|
|
|
func _input(event):
|
|
|
|
if mouse_inside:
|
2020-09-22 13:15:36 +02:00
|
|
|
if event.is_action_pressed("zoom_out") and nodes.rect_scale.x > 0.3:
|
|
|
|
nodes.rect_scale -= Vector2(0.05, 0.05)
|
|
|
|
if event.is_action_pressed("zoom_in") and nodes.rect_scale.x < 2:
|
|
|
|
nodes.rect_scale += Vector2(0.05, 0.05)
|
2020-09-21 19:28:39 +02:00
|
|
|
|
|
|
|
func there_is_a_git():
|
|
|
|
return shell.run("test -d .git && echo yes || echo no") == "yes\n"
|
|
|
|
|
2020-09-01 19:20:51 +02:00
|
|
|
func update_everything():
|
2020-09-30 21:36:11 +02:00
|
|
|
if file_browser:
|
|
|
|
file_browser.update()
|
2020-09-21 19:28:39 +02:00
|
|
|
if there_is_a_git():
|
2020-09-16 16:16:46 +02:00
|
|
|
update_head()
|
|
|
|
update_refs()
|
|
|
|
update_index()
|
|
|
|
update_objects()
|
|
|
|
remove_gone_stuff()
|
2020-09-18 11:15:09 +02:00
|
|
|
else:
|
2020-09-30 21:36:11 +02:00
|
|
|
if index:
|
|
|
|
index.text = ""
|
2020-09-22 21:56:22 +02:00
|
|
|
for o in objects:
|
|
|
|
objects[o].queue_free()
|
2020-09-29 17:20:24 +02:00
|
|
|
objects = {}
|
2020-09-01 17:24:21 +02:00
|
|
|
|
|
|
|
func set_path(new_path):
|
|
|
|
path = new_path
|
2020-09-30 21:36:11 +02:00
|
|
|
if path_node:
|
|
|
|
path_node.text = path
|
2020-10-01 10:07:36 +02:00
|
|
|
if new_path != "":
|
|
|
|
shell.cd(new_path)
|
|
|
|
for o in objects.values():
|
|
|
|
o.queue_free()
|
|
|
|
objects = {}
|
|
|
|
if is_inside_tree():
|
|
|
|
update_everything()
|
2020-09-01 17:24:21 +02:00
|
|
|
|
|
|
|
func get_path():
|
|
|
|
return path
|
|
|
|
|
2020-09-03 18:10:09 +02:00
|
|
|
func set_label(new_label):
|
|
|
|
label = new_label
|
2020-09-22 13:15:36 +02:00
|
|
|
if label_node:
|
|
|
|
label_node.text = new_label
|
2020-09-03 18:10:09 +02:00
|
|
|
|
2020-09-01 17:24:21 +02:00
|
|
|
func update_index():
|
2020-09-22 13:15:36 +02:00
|
|
|
index.text = git("ls-files -s --abbrev=8").replace("\t", " ")
|
2020-09-01 17:24:21 +02:00
|
|
|
|
|
|
|
func random_position():
|
2020-09-03 17:50:03 +02:00
|
|
|
return Vector2(rand_range(0, rect_size.x), rand_range(0, rect_size.y))
|
2020-09-01 17:24:21 +02:00
|
|
|
|
|
|
|
func update_objects():
|
2020-09-15 16:59:54 +02:00
|
|
|
var all = all_objects()
|
|
|
|
|
|
|
|
# Create new objects, if necessary.
|
|
|
|
for o in all:
|
2020-09-01 17:24:21 +02:00
|
|
|
if objects.has(o):
|
|
|
|
continue
|
2020-09-14 16:03:01 +02:00
|
|
|
|
|
|
|
var type = object_type(o)
|
2020-09-23 12:05:43 +02:00
|
|
|
if simplified_view:
|
2020-09-14 16:03:01 +02:00
|
|
|
if type == "tree" or type == "blob":
|
|
|
|
continue
|
|
|
|
|
2020-09-01 17:24:21 +02:00
|
|
|
var n = node.instance()
|
|
|
|
n.id = o
|
|
|
|
n.type = object_type(o)
|
|
|
|
n.content = object_content(o)
|
|
|
|
n.repository = self
|
|
|
|
|
2020-09-14 16:03:01 +02:00
|
|
|
match type:
|
|
|
|
"blob":
|
|
|
|
pass
|
|
|
|
"tree":
|
|
|
|
n.children = tree_children(o)
|
2020-09-15 22:34:22 +02:00
|
|
|
n.content = n.content.replacen("\t", " ")
|
2020-09-14 16:03:01 +02:00
|
|
|
"commit":
|
|
|
|
var c = {}
|
|
|
|
c[commit_tree(o)] = ""
|
|
|
|
for p in commit_parents(o):
|
|
|
|
c[p] = ""
|
|
|
|
n.children = c
|
2020-09-30 22:41:28 +02:00
|
|
|
|
|
|
|
_commit_count += 1
|
|
|
|
if _commit_count >= 3 and not simplified_view:
|
|
|
|
simplified_view = true
|
2020-09-14 16:03:01 +02:00
|
|
|
"tag":
|
|
|
|
n.children = tag_target(o)
|
2020-09-14 14:54:30 +02:00
|
|
|
|
|
|
|
n.position = find_position(n)
|
2020-09-22 13:15:36 +02:00
|
|
|
nodes.add_child(n)
|
2020-09-01 17:24:21 +02:00
|
|
|
objects[o] = n
|
2020-10-01 13:11:29 +02:00
|
|
|
|
|
|
|
func update_node_positions():
|
|
|
|
if there_is_a_git():
|
|
|
|
var graph_text = shell.run("git log --graph --oneline --all --no-abbrev")
|
|
|
|
var graph_lines = Array(graph_text.split("\n"))
|
|
|
|
graph_lines.pop_back()
|
|
|
|
|
|
|
|
for line_count in range(graph_lines.size()):
|
|
|
|
var line = graph_lines[line_count]
|
|
|
|
if "*" in line:
|
|
|
|
var star_idx = line.find("*")
|
|
|
|
var hash_regex = RegEx.new()
|
|
|
|
hash_regex.compile("[a-f0-9]+")
|
|
|
|
var regex_match = hash_regex.search(line)
|
|
|
|
objects[regex_match.get_string()].position = Vector2(star_idx * 100 + 500, line_count * 100 + 500)
|
|
|
|
|
|
|
|
for ref in all_refs():
|
|
|
|
var target_reference = objects[ref].children.keys()[0]
|
|
|
|
var target = objects[target_reference]
|
|
|
|
objects[ref].position = Vector2(target.position.x ,target.position.y - 100)
|
|
|
|
|
|
|
|
var target_reference = objects["HEAD"].children.keys()[0]
|
2020-10-01 14:56:24 +02:00
|
|
|
if objects.has(target_reference):
|
|
|
|
var target = objects[target_reference]
|
|
|
|
objects["HEAD"].position = Vector2(target.position.x ,target.position.y - 100)
|
2020-10-01 13:11:29 +02:00
|
|
|
|
|
|
|
|
|
|
|
|
2020-09-01 17:24:21 +02:00
|
|
|
|
|
|
|
func update_refs():
|
|
|
|
for r in all_refs():
|
|
|
|
if not objects.has(r):
|
|
|
|
var n = node.instance()
|
|
|
|
n.id = r
|
|
|
|
n.type = "ref"
|
|
|
|
n.content = ""
|
|
|
|
n.repository = self
|
|
|
|
objects[r] = n
|
2020-09-14 14:54:30 +02:00
|
|
|
n.children = {ref_target(r): ""}
|
|
|
|
n.position = find_position(n)
|
2020-09-22 13:15:36 +02:00
|
|
|
nodes.add_child(n)
|
2020-09-01 17:24:21 +02:00
|
|
|
var n = objects[r]
|
2020-09-13 21:55:24 +02:00
|
|
|
n.children = {ref_target(r): ""}
|
2020-09-01 17:24:21 +02:00
|
|
|
|
|
|
|
func apply_forces():
|
|
|
|
for o in objects.values():
|
2020-09-14 16:03:01 +02:00
|
|
|
if not o.visible:
|
|
|
|
continue
|
2020-09-01 17:24:21 +02:00
|
|
|
for o2 in objects.values():
|
2020-09-14 16:03:01 +02:00
|
|
|
if o == o2 or not o2.visible:
|
2020-09-01 17:24:21 +02:00
|
|
|
continue
|
|
|
|
var d = o.position.distance_to(o2.position)
|
|
|
|
var dir = (o.global_position - o2.global_position).normalized()
|
2020-09-14 14:54:30 +02:00
|
|
|
var f = 2000/pow(d+0.00001,1.5)
|
2020-09-01 17:24:21 +02:00
|
|
|
o.position += dir*f
|
|
|
|
o2.position -= dir*f
|
2020-09-24 10:31:41 +02:00
|
|
|
var center_of_gravity = nodes.rect_size/2
|
2020-09-03 17:50:03 +02:00
|
|
|
var d = o.position.distance_to(center_of_gravity)
|
|
|
|
var dir = (o.position - center_of_gravity).normalized()
|
2020-10-01 13:11:29 +02:00
|
|
|
var f = (d+0.00001)*(Vector2(nodes.rect_size.y, nodes.rect_size.x/3).normalized()/30)
|
2020-09-01 17:24:21 +02:00
|
|
|
o.position -= dir*f
|
2020-09-14 14:54:30 +02:00
|
|
|
|
|
|
|
func find_position(n):
|
|
|
|
var position = Vector2.ZERO
|
|
|
|
var count = 0
|
|
|
|
for child in n.children:
|
|
|
|
if objects.has(child):
|
|
|
|
position += objects[child].position
|
|
|
|
count += 1
|
|
|
|
if count > 0:
|
|
|
|
position /= count
|
|
|
|
n.position = position + Vector2(0, -150)
|
|
|
|
else:
|
|
|
|
n.position = random_position()
|
|
|
|
return n.position
|
2020-09-01 17:24:21 +02:00
|
|
|
|
|
|
|
func git(args, splitlines = false):
|
2020-09-22 21:56:22 +02:00
|
|
|
var o = shell.run("git --no-replace-objects " + args)
|
2020-09-01 17:24:21 +02:00
|
|
|
|
|
|
|
if splitlines:
|
|
|
|
o = o.split("\n")
|
|
|
|
# Remove last empty line.
|
|
|
|
o.remove(len(o)-1)
|
|
|
|
else:
|
|
|
|
# Remove trailing newline.
|
|
|
|
o = o.substr(0,len(o)-1)
|
|
|
|
return o
|
|
|
|
|
|
|
|
func update_head():
|
|
|
|
if not objects.has("HEAD"):
|
|
|
|
var n = node.instance()
|
|
|
|
n.id = "HEAD"
|
|
|
|
n.type = "head"
|
|
|
|
n.content = ""
|
|
|
|
n.repository = self
|
2020-09-14 14:54:30 +02:00
|
|
|
n.position = find_position(n)
|
|
|
|
|
2020-09-01 17:24:21 +02:00
|
|
|
objects["HEAD"] = n
|
2020-09-22 13:15:36 +02:00
|
|
|
nodes.add_child(n)
|
2020-09-01 17:24:21 +02:00
|
|
|
var n = objects["HEAD"]
|
2020-09-13 21:55:24 +02:00
|
|
|
n.children = {ref_target("HEAD"): ""}
|
2020-09-01 17:24:21 +02:00
|
|
|
|
|
|
|
func all_objects():
|
2020-09-16 16:16:46 +02:00
|
|
|
var obj = git("cat-file --batch-check='%(objectname)' --batch-all-objects", true)
|
2020-09-15 16:59:54 +02:00
|
|
|
var dict = {}
|
2020-09-16 16:16:46 +02:00
|
|
|
for o in obj:
|
2020-09-15 16:59:54 +02:00
|
|
|
dict[o] = ""
|
|
|
|
return dict
|
2020-09-01 17:24:21 +02:00
|
|
|
|
|
|
|
func object_type(id):
|
|
|
|
return git("cat-file -t "+id)
|
|
|
|
|
|
|
|
func object_content(id):
|
|
|
|
return git("cat-file -p "+id)
|
|
|
|
|
|
|
|
func tree_children(id):
|
|
|
|
var children = git("cat-file -p "+id, true)
|
|
|
|
var ids = {}
|
|
|
|
for c in children:
|
|
|
|
var a = c.split(" ")
|
|
|
|
ids[a[2].split("\t")[0]] = a[2].split("\t")[1]
|
|
|
|
return ids
|
|
|
|
|
|
|
|
func commit_tree(id):
|
|
|
|
var c = git("cat-file -p "+id, true)
|
|
|
|
for cc in c:
|
|
|
|
var ccc = cc.split(" ", 2)
|
|
|
|
match ccc[0]:
|
|
|
|
"tree":
|
|
|
|
return ccc[1]
|
|
|
|
return null
|
|
|
|
|
|
|
|
func commit_parents(id):
|
|
|
|
var parents = []
|
|
|
|
var c = git("cat-file -p "+id, true)
|
|
|
|
for cc in c:
|
|
|
|
var ccc = cc.split(" ", 2)
|
|
|
|
match ccc[0]:
|
|
|
|
"parent":
|
|
|
|
parents.push_back(ccc[1])
|
|
|
|
return parents
|
|
|
|
|
2020-09-13 21:55:24 +02:00
|
|
|
func tag_target(id):
|
|
|
|
var c = git("rev-parse %s^{}" % id)
|
|
|
|
return {c: ""}
|
|
|
|
|
2020-09-01 17:24:21 +02:00
|
|
|
func all_refs():
|
2020-09-15 16:59:54 +02:00
|
|
|
var refs = {}
|
2020-09-08 20:26:14 +02:00
|
|
|
# If there are no refs, show-ref will have exit code 1. We don't care.
|
|
|
|
for line in git("show-ref || true", true):
|
2020-09-01 17:24:21 +02:00
|
|
|
line = line.split(" ")
|
2020-09-08 20:26:14 +02:00
|
|
|
var _id = line[0]
|
2020-09-01 17:24:21 +02:00
|
|
|
var name = line[1]
|
2020-09-15 16:59:54 +02:00
|
|
|
refs[name] = ""
|
2020-09-01 17:24:21 +02:00
|
|
|
return refs
|
|
|
|
|
2020-09-13 21:55:24 +02:00
|
|
|
func ref_target(ref):
|
|
|
|
# Test whether this is a symbolic ref.
|
|
|
|
var ret = git("symbolic-ref -q "+ref+" || true")
|
|
|
|
# If it's not, it's probably a regular ref.
|
|
|
|
if ret == "":
|
2020-09-14 15:35:30 +02:00
|
|
|
if ref == "HEAD":
|
|
|
|
ret = git("show-ref --head "+ref).split(" ")[0]
|
|
|
|
else:
|
|
|
|
ret = git("show-ref "+ref).split(" ")[0]
|
2020-09-13 21:55:24 +02:00
|
|
|
return ret
|
2020-09-14 16:03:01 +02:00
|
|
|
|
2020-09-30 22:41:28 +02:00
|
|
|
func set_simplified_view(simplify):
|
|
|
|
simplified_view = simplify
|
|
|
|
if simplify_checkbox:
|
|
|
|
simplify_checkbox.pressed = simplify
|
|
|
|
|
2020-09-14 16:03:01 +02:00
|
|
|
for o in objects:
|
|
|
|
var obj = objects[o]
|
|
|
|
if obj.type == "tree" or obj.type == "blob":
|
2020-09-30 22:41:28 +02:00
|
|
|
obj.visible = not simplify
|
2020-09-14 16:03:01 +02:00
|
|
|
|
2020-09-21 19:28:39 +02:00
|
|
|
if there_is_a_git():
|
|
|
|
update_objects()
|
2020-09-15 16:59:54 +02:00
|
|
|
|
2020-09-29 17:20:24 +02:00
|
|
|
func set_editable_path(editable):
|
|
|
|
editable_path = editable
|
|
|
|
if label_node:
|
|
|
|
label_node.visible = not editable
|
|
|
|
if path_node:
|
|
|
|
path_node.visible = editable
|
|
|
|
|
2020-09-15 16:59:54 +02:00
|
|
|
func remove_gone_stuff():
|
|
|
|
# FIXME: Cache the result of all_objects.
|
|
|
|
var all = {}
|
|
|
|
for o in all_objects():
|
|
|
|
all[o] = ""
|
|
|
|
for o in all_refs():
|
|
|
|
all[o] = ""
|
|
|
|
all["HEAD"] = ""
|
|
|
|
# Delete objects, if they disappeared.
|
2020-09-15 17:23:22 +02:00
|
|
|
for o in objects.keys():
|
2020-09-15 16:59:54 +02:00
|
|
|
if not all.has(o):
|
|
|
|
objects[o].queue_free()
|
|
|
|
objects.erase(o)
|
2020-09-15 18:41:06 +02:00
|
|
|
|
|
|
|
func _on_mouse_entered():
|
|
|
|
mouse_inside = true
|
|
|
|
|
|
|
|
func _on_mouse_exited():
|
|
|
|
mouse_inside = false
|
2020-09-22 13:15:36 +02:00
|
|
|
|
|
|
|
func set_file_browser_active(active):
|
|
|
|
file_browser_active = active
|
|
|
|
if file_browser:
|
|
|
|
file_browser.visible = active
|
|
|
|
|
|
|
|
|