- New selfmade Font
- persistent files - Hiscore
31
.github/ISSUE_TEMPLATE/bug_report.md
vendored
Normal file
|
@ -0,0 +1,31 @@
|
||||||
|
---
|
||||||
|
name: Bug report
|
||||||
|
about: Create a report to help us improve
|
||||||
|
title: ''
|
||||||
|
labels: bug
|
||||||
|
assignees: Ericdowney
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
**Describe the bug**
|
||||||
|
A clear and concise description of what the bug is.
|
||||||
|
|
||||||
|
**To Reproduce**
|
||||||
|
Steps to reproduce the behavior:
|
||||||
|
1. Go to '...'
|
||||||
|
2. Click on '....'
|
||||||
|
3. Scroll down to '....'
|
||||||
|
4. See error
|
||||||
|
|
||||||
|
**Expected behavior**
|
||||||
|
A clear and concise description of what you expected to happen.
|
||||||
|
|
||||||
|
**Screenshots**
|
||||||
|
If applicable, add screenshots to help explain your problem.
|
||||||
|
|
||||||
|
**Desktop (please complete the following information):**
|
||||||
|
- OS: [e.g. macOS]
|
||||||
|
- Godot Version [e.g. 4.1.1 stable]
|
||||||
|
|
||||||
|
**Additional context**
|
||||||
|
Add any other context about the problem here.
|
21
LICENSE
Normal file
|
@ -0,0 +1,21 @@
|
||||||
|
MIT License
|
||||||
|
|
||||||
|
Copyright (c) 2023 Ericdowney
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
of this software and associated documentation files (the "Software"), to deal
|
||||||
|
in the Software without restriction, including without limitation the rights
|
||||||
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
copies of the Software, and to permit persons to whom the Software is
|
||||||
|
furnished to do so, subject to the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be included in all
|
||||||
|
copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
SOFTWARE.
|
BIN
Megamichisfont-small.otf
Normal file
|
@ -2,13 +2,13 @@
|
||||||
|
|
||||||
importer="font_data_dynamic"
|
importer="font_data_dynamic"
|
||||||
type="FontFile"
|
type="FontFile"
|
||||||
uid="uid://da6fxih8qew6g"
|
uid="uid://ddvhypxkj8a1b"
|
||||||
path="res://.godot/imported/m5x7.ttf-ede74fbe3753eb2fe0ebc455b7b3a4db.fontdata"
|
path="res://.godot/imported/Megamichisfont-small.otf-43fae38c860095c22170042e51803feb.fontdata"
|
||||||
|
|
||||||
[deps]
|
[deps]
|
||||||
|
|
||||||
source_file="res://m5x7.ttf"
|
source_file="res://Megamichisfont-small.otf"
|
||||||
dest_files=["res://.godot/imported/m5x7.ttf-ede74fbe3753eb2fe0ebc455b7b3a4db.fontdata"]
|
dest_files=["res://.godot/imported/Megamichisfont-small.otf-43fae38c860095c22170042e51803feb.fontdata"]
|
||||||
|
|
||||||
[params]
|
[params]
|
||||||
|
|
1
addons/SignalVisualizer/Clear.svg
Normal file
|
@ -0,0 +1 @@
|
||||||
|
<svg height="24" viewBox="0 0 16 16" width="24" xmlns="http://www.w3.org/2000/svg"><path d="m8 1a1 1 0 0 0 -1 1v5h-2c-1.108 0-2 .892-2 2v1h10v-1c0-1.108-.892-2-2-2h-2v-5a1 1 0 0 0 -1-1zm-5 10v4l10-1v-3z" fill="#e0e0e0"/></svg>
|
After Width: | Height: | Size: 227 B |
37
addons/SignalVisualizer/Clear.svg.import
Normal file
|
@ -0,0 +1,37 @@
|
||||||
|
[remap]
|
||||||
|
|
||||||
|
importer="texture"
|
||||||
|
type="CompressedTexture2D"
|
||||||
|
uid="uid://bmnff63evbdhv"
|
||||||
|
path="res://.godot/imported/Clear.svg-d661617e27b91e3580171e3447fde514.ctex"
|
||||||
|
metadata={
|
||||||
|
"vram_texture": false
|
||||||
|
}
|
||||||
|
|
||||||
|
[deps]
|
||||||
|
|
||||||
|
source_file="res://addons/SignalVisualizer/Clear.svg"
|
||||||
|
dest_files=["res://.godot/imported/Clear.svg-d661617e27b91e3580171e3447fde514.ctex"]
|
||||||
|
|
||||||
|
[params]
|
||||||
|
|
||||||
|
compress/mode=0
|
||||||
|
compress/high_quality=false
|
||||||
|
compress/lossy_quality=0.7
|
||||||
|
compress/hdr_compression=1
|
||||||
|
compress/normal_map=0
|
||||||
|
compress/channel_pack=0
|
||||||
|
mipmaps/generate=false
|
||||||
|
mipmaps/limit=-1
|
||||||
|
roughness/mode=0
|
||||||
|
roughness/src_normal=""
|
||||||
|
process/fix_alpha_border=true
|
||||||
|
process/premult_alpha=false
|
||||||
|
process/normal_map_invert_y=false
|
||||||
|
process/hdr_as_srgb=false
|
||||||
|
process/hdr_clamp_exposure=false
|
||||||
|
process/size_limit=0
|
||||||
|
detect_3d/compress_to=1
|
||||||
|
svg/scale=1.0
|
||||||
|
editor/scale_with_editor_scale=false
|
||||||
|
editor/convert_colors_with_editor_theme=false
|
58
addons/SignalVisualizer/Common/signal_connection.gd
Normal file
|
@ -0,0 +1,58 @@
|
||||||
|
class_name SignalConnection extends Object
|
||||||
|
|
||||||
|
# Properties
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
|
||||||
|
var signal_id: int
|
||||||
|
var source_node_name: String
|
||||||
|
var destination_node_name: String
|
||||||
|
var method_signature: String
|
||||||
|
|
||||||
|
var description: String :
|
||||||
|
get:
|
||||||
|
return "ID: {signal_id} Source: {source_node_name} Destination: {destination_node_name} Method: {method_signature}".format({
|
||||||
|
"signal_id": signal_id,
|
||||||
|
"source_node_name": source_node_name,
|
||||||
|
"destination_node_name": destination_node_name,
|
||||||
|
"method_signature": method_signature,
|
||||||
|
})
|
||||||
|
|
||||||
|
var dictionary_key: String :
|
||||||
|
get:
|
||||||
|
return "{signal_id}__{source_node_name}__{destination_node_name}__{method_signature}".format({ "signal_id": signal_id, "source_node_name": source_node_name, "destination_node_name": destination_node_name, "method_signature": method_signature.replace("::", "_") })
|
||||||
|
|
||||||
|
var dictionary_representation: Dictionary :
|
||||||
|
get:
|
||||||
|
return {
|
||||||
|
"signal_id": signal_id,
|
||||||
|
"source_node_name": source_node_name,
|
||||||
|
"destination_node_name": destination_node_name,
|
||||||
|
"method_signature": method_signature,
|
||||||
|
}
|
||||||
|
|
||||||
|
# Lifecycle
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
|
||||||
|
func _init(signal_id: int, source_node_name: String, destination_node_name: String, method_signature: String):
|
||||||
|
self.signal_id = signal_id
|
||||||
|
self.source_node_name = source_node_name
|
||||||
|
self.destination_node_name = destination_node_name
|
||||||
|
self.method_signature = method_signature
|
||||||
|
|
||||||
|
# Signals
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
# Methods
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
|
||||||
|
|
56
addons/SignalVisualizer/Common/signal_description.gd
Normal file
|
@ -0,0 +1,56 @@
|
||||||
|
class_name SignalDescription extends Object
|
||||||
|
|
||||||
|
# Properties
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
|
||||||
|
var id: int:
|
||||||
|
get:
|
||||||
|
if _source_id != null:
|
||||||
|
return _source_id
|
||||||
|
return get_instance_id()
|
||||||
|
|
||||||
|
var node_name: String
|
||||||
|
var signal_name: String
|
||||||
|
|
||||||
|
var description: String :
|
||||||
|
get:
|
||||||
|
return "ID: {id} Node: {node_name} Signal: {signal_name}".format({
|
||||||
|
"id": id,
|
||||||
|
"node_name": node_name,
|
||||||
|
"signal_name": signal_name,
|
||||||
|
})
|
||||||
|
|
||||||
|
var dictionary_representation: Dictionary :
|
||||||
|
get:
|
||||||
|
return {
|
||||||
|
"id": id,
|
||||||
|
"node_name": node_name,
|
||||||
|
"signal_name": signal_name,
|
||||||
|
}
|
||||||
|
|
||||||
|
var _source_id = null
|
||||||
|
|
||||||
|
# Lifecycle
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
|
||||||
|
func _init(node_name: String, signal_name: String):
|
||||||
|
self.node_name = node_name
|
||||||
|
self.signal_name = signal_name
|
||||||
|
|
||||||
|
# Signals
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
# Methods
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
|
||||||
|
|
53
addons/SignalVisualizer/Common/signal_graph.gd
Normal file
|
@ -0,0 +1,53 @@
|
||||||
|
class_name SignalGraph extends Object
|
||||||
|
|
||||||
|
# Properties
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
|
||||||
|
var name: String
|
||||||
|
var signals: Array[SignalDescription]
|
||||||
|
var edges: Array[SignalConnection]
|
||||||
|
|
||||||
|
var description: String :
|
||||||
|
get:
|
||||||
|
return "Signals: {signals}\nEdges: {edges}".format({
|
||||||
|
"signals": signals.map(func (item): return item.description),
|
||||||
|
"edges": edges.map(func (item): return item.description),
|
||||||
|
})
|
||||||
|
|
||||||
|
var dictionary_representation: Dictionary :
|
||||||
|
get:
|
||||||
|
return {
|
||||||
|
"name": name,
|
||||||
|
"signals": signals.map(func (element): return element.dictionary_representation),
|
||||||
|
"edges": edges.map(func (element): return element.dictionary_representation),
|
||||||
|
}
|
||||||
|
|
||||||
|
# Lifecycle
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
|
||||||
|
func _init(name: String, signals: Array[SignalDescription] = [], edges: Array[SignalConnection] = []):
|
||||||
|
self.name = name
|
||||||
|
self.signals = signals
|
||||||
|
self.edges = edges
|
||||||
|
|
||||||
|
# Signals
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
# Methods
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
|
||||||
|
func get_source_signal_for_edge(edge: SignalConnection) -> SignalDescription:
|
||||||
|
var result = signals.filter(func (item): return item.id == edge.signal_id)
|
||||||
|
if result.size() > 0:
|
||||||
|
return result[0]
|
||||||
|
return null
|
170
addons/SignalVisualizer/Common/signal_graph_utility.gd
Normal file
|
@ -0,0 +1,170 @@
|
||||||
|
@tool
|
||||||
|
class_name SignalGraphUtility
|
||||||
|
|
||||||
|
static var SignalGraphNode = preload("res://addons/SignalVisualizer/Visualizer/signal_graph_node.tscn")
|
||||||
|
static var GraphNodeItem = preload("res://addons/SignalVisualizer/Visualizer/signal_graph_node_item.tscn")
|
||||||
|
|
||||||
|
const SOURCE_COLOR: Color = Color.SKY_BLUE
|
||||||
|
const DESTINATION_COLOR: Color = Color.CORAL
|
||||||
|
const CONNECTION_TYPE: int = 0
|
||||||
|
|
||||||
|
#region Methods
|
||||||
|
|
||||||
|
static func create_signal_graph(name: String, signals: Array, edges: Array) -> SignalGraph:
|
||||||
|
var signal_graph = SignalGraph.new(name)
|
||||||
|
|
||||||
|
for signal_item in signals:
|
||||||
|
var new_signal_description = SignalDescription.new(signal_item.node_name, signal_item.signal_name)
|
||||||
|
new_signal_description._source_id = signal_item.id
|
||||||
|
signal_graph.signals.append(new_signal_description)
|
||||||
|
|
||||||
|
for connection in edges:
|
||||||
|
var new_edge = SignalConnection.new(connection.signal_id, connection.source_node_name, connection.destination_node_name, connection.method_signature)
|
||||||
|
signal_graph.edges.append(new_edge)
|
||||||
|
|
||||||
|
return signal_graph
|
||||||
|
|
||||||
|
static func create_signal_graph_from_node(root_node: Node, is_persistent_only: bool = false):
|
||||||
|
var signal_graph = SignalGraph.new(root_node.scene_file_path)
|
||||||
|
var all_nodes: Array[Node] = _gather_nodes_from_node(root_node)
|
||||||
|
var signals: Array[SignalDescription] = []
|
||||||
|
var edges: Array[SignalConnection] = []
|
||||||
|
|
||||||
|
for node in all_nodes:
|
||||||
|
for signal_item in node.get_signal_list():
|
||||||
|
var existing_signals = []
|
||||||
|
var connection_list = node.get_signal_connection_list(signal_item["name"] as String)
|
||||||
|
if connection_list.size() > 0:
|
||||||
|
for connection in connection_list:
|
||||||
|
var enabled_flags = connection["flags"] == CONNECT_PERSIST if is_persistent_only else true
|
||||||
|
var should_display_connection = "name" in connection["callable"].get_object() and not connection["callable"].get_object().name.begins_with("@") and enabled_flags
|
||||||
|
if should_display_connection:
|
||||||
|
var signal_description: SignalDescription
|
||||||
|
var filtered_signals = existing_signals.filter(func (element): return element.signal_name == signal_item.name and element.node_name == node.name)
|
||||||
|
if filtered_signals.size() == 1:
|
||||||
|
signal_description = filtered_signals[0]
|
||||||
|
else:
|
||||||
|
signal_description = SignalDescription.new(node.name, signal_item.name)
|
||||||
|
existing_signals.append(signal_description)
|
||||||
|
signals.append(signal_description)
|
||||||
|
|
||||||
|
var signal_edge = SignalConnection.new(signal_description.id, signal_description.node_name, connection["callable"].get_object().name, connection["callable"].get_method())
|
||||||
|
if not signal_graph.edges.any(func (element): return element.signal_id == signal_description.id):
|
||||||
|
edges.append(signal_edge)
|
||||||
|
|
||||||
|
var temp_signals = {}
|
||||||
|
for item in signals:
|
||||||
|
temp_signals[item.id] = item
|
||||||
|
|
||||||
|
var temp_edges = {}
|
||||||
|
for item in edges:
|
||||||
|
temp_edges[item.dictionary_key] = item
|
||||||
|
|
||||||
|
signal_graph.signals.assign(temp_signals.keys().map(func (key): return temp_signals[key]))
|
||||||
|
signal_graph.edges.assign(temp_edges.keys().map(func (key): return temp_edges[key]))
|
||||||
|
|
||||||
|
return signal_graph
|
||||||
|
|
||||||
|
static func generate_signal_graph_nodes(signal_graph: SignalGraph, graph_node: GraphEdit, open_script_callable: Callable):
|
||||||
|
var graph_nodes: Dictionary = {}
|
||||||
|
|
||||||
|
for signal_item in signal_graph.signals:
|
||||||
|
var current_graph_node: SignalGraphNode
|
||||||
|
if graph_nodes.has(signal_item.node_name):
|
||||||
|
current_graph_node = graph_nodes[signal_item.node_name]
|
||||||
|
if not current_graph_node:
|
||||||
|
current_graph_node = SignalGraphNode.instantiate()
|
||||||
|
current_graph_node.title = signal_item.node_name
|
||||||
|
current_graph_node.name = _get_graph_node_name(signal_item.node_name)
|
||||||
|
graph_node.add_child(current_graph_node)
|
||||||
|
graph_nodes[signal_item.node_name] = current_graph_node
|
||||||
|
|
||||||
|
for edge in signal_graph.edges:
|
||||||
|
var destination_graph_node: SignalGraphNode
|
||||||
|
if graph_nodes.has(edge.destination_node_name):
|
||||||
|
destination_graph_node = graph_nodes[edge.destination_node_name]
|
||||||
|
else:
|
||||||
|
destination_graph_node = SignalGraphNode.instantiate()
|
||||||
|
destination_graph_node.title = edge.destination_node_name
|
||||||
|
destination_graph_node.name = _get_graph_node_name(edge.destination_node_name)
|
||||||
|
graph_node.add_child(destination_graph_node)
|
||||||
|
graph_nodes[edge.destination_node_name] = destination_graph_node
|
||||||
|
|
||||||
|
var source_signal = signal_graph.get_source_signal_for_edge(edge)
|
||||||
|
if source_signal != null:
|
||||||
|
var source_graph_node: SignalGraphNode = graph_nodes[edge.source_node_name] as SignalGraphNode
|
||||||
|
|
||||||
|
if not source_graph_node.has_source_signal_description(source_signal.signal_name, edge.destination_node_name):
|
||||||
|
var source_signal_label = Label.new()
|
||||||
|
source_signal_label.text = source_signal.signal_name
|
||||||
|
source_signal_label.name = "source_" + source_signal.signal_name + "_" + edge.destination_node_name
|
||||||
|
source_graph_node.add_child(source_signal_label)
|
||||||
|
|
||||||
|
var destination_signal_name = "destination_" + source_signal.signal_name + "_" + edge.method_signature.replace("::", "__")
|
||||||
|
var has_destination = destination_graph_node.has_destination_signal_description(source_signal.signal_name, edge.method_signature)
|
||||||
|
if not has_destination:
|
||||||
|
var destination_signal_item = GraphNodeItem.instantiate()
|
||||||
|
destination_signal_item.signal_data = SignalGraphNodeItem.Metadata.new(source_signal.signal_name, edge.method_signature, edge.destination_node_name)
|
||||||
|
destination_signal_item.text = edge.method_signature
|
||||||
|
destination_signal_item.name = destination_signal_name
|
||||||
|
destination_signal_item.open_script.connect(open_script_callable)
|
||||||
|
destination_graph_node.add_child(destination_signal_item)
|
||||||
|
|
||||||
|
for edge in signal_graph.edges:
|
||||||
|
var source_signal = signal_graph.get_source_signal_for_edge(edge)
|
||||||
|
if source_signal != null:
|
||||||
|
var source_graph_node: SignalGraphNode = graph_nodes[edge.source_node_name] as SignalGraphNode
|
||||||
|
var destination_graph_node: SignalGraphNode = graph_nodes[edge.destination_node_name] as SignalGraphNode
|
||||||
|
|
||||||
|
var from_port = source_graph_node.get_source_slot(source_signal.signal_name, edge.destination_node_name)
|
||||||
|
var to_port = destination_graph_node.get_destination_slot(source_signal.signal_name, edge.method_signature)
|
||||||
|
|
||||||
|
source_graph_node.set_slot(from_port, false, CONNECTION_TYPE, Color.BLACK, true, CONNECTION_TYPE, SOURCE_COLOR)
|
||||||
|
destination_graph_node.set_slot(to_port, true, CONNECTION_TYPE, DESTINATION_COLOR, false, CONNECTION_TYPE, Color.BLACK)
|
||||||
|
|
||||||
|
var from_slot_index = source_graph_node.get_next_source_slot(source_signal.signal_name, edge.destination_node_name)
|
||||||
|
var to_slot_index = destination_graph_node.get_next_destination_slot(source_signal.signal_name, edge.method_signature)
|
||||||
|
|
||||||
|
if from_port >= 0 and to_port >= 0:
|
||||||
|
graph_node.connect_node(source_graph_node.name, from_slot_index, destination_graph_node.name, to_slot_index)
|
||||||
|
else:
|
||||||
|
print(">>> Invalid Connection Request")
|
||||||
|
|
||||||
|
static func generate_signal_graph_tree(signal_graph: SignalGraph, tree_node: Tree):
|
||||||
|
var root = tree_node.create_item()
|
||||||
|
root.set_text(0, signal_graph.name)
|
||||||
|
|
||||||
|
var tree_items: Dictionary = {}
|
||||||
|
|
||||||
|
for signal_item in signal_graph.signals:
|
||||||
|
var node_tree_item: TreeItem
|
||||||
|
if tree_items.has(signal_item.node_name):
|
||||||
|
node_tree_item = tree_items[signal_item.node_name] as TreeItem
|
||||||
|
else:
|
||||||
|
node_tree_item = tree_node.create_item(root)
|
||||||
|
node_tree_item.set_text(0, signal_item.node_name)
|
||||||
|
tree_items[signal_item.node_name] = node_tree_item
|
||||||
|
|
||||||
|
var signal_tree_item = tree_node.create_item(node_tree_item)
|
||||||
|
signal_tree_item.set_text(0, signal_item.signal_name)
|
||||||
|
|
||||||
|
for edge in signal_graph.edges.filter(func (item): return item.signal_id == signal_item.id):
|
||||||
|
var signal_connection_tree_item = tree_node.create_item(signal_tree_item)
|
||||||
|
signal_connection_tree_item.set_text(0, edge.destination_node_name + "::" + edge.method_signature)
|
||||||
|
|
||||||
|
static func _get_graph_node_name(name: String) -> String:
|
||||||
|
return "{node_name}_graph_node".format({ "node_name": name })
|
||||||
|
|
||||||
|
static func _gather_nodes_from_node(root_node: Node) -> Array[Node]:
|
||||||
|
var node_list: Array[Node] = [root_node]
|
||||||
|
return node_list + __gather_nodes_from_node(root_node)
|
||||||
|
|
||||||
|
static func __gather_nodes_from_node(node: Node) -> Array[Node]:
|
||||||
|
var nodes: Array[Node] = []
|
||||||
|
for child in node.get_children(false):
|
||||||
|
nodes.append(child)
|
||||||
|
nodes += __gather_nodes_from_node(child)
|
||||||
|
|
||||||
|
return nodes
|
||||||
|
|
||||||
|
#endregion
|
146
addons/SignalVisualizer/Debugger/SignalDebugger.gd
Normal file
|
@ -0,0 +1,146 @@
|
||||||
|
extends Node
|
||||||
|
|
||||||
|
# Properties
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
|
||||||
|
var _signal_graph: SignalGraph
|
||||||
|
var _lambda_map: Dictionary = {}
|
||||||
|
|
||||||
|
# Lifecycle
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
|
||||||
|
func _ready():
|
||||||
|
if OS.is_debug_build():
|
||||||
|
EngineDebugger.register_message_capture("signal_debugger", _on_signal_debugger_message_capture)
|
||||||
|
|
||||||
|
# Signals
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
|
||||||
|
func _on_signal_debugger_message_capture(message: String, data: Array) -> bool:
|
||||||
|
if message == "start":
|
||||||
|
_signal_graph = generate_signal_graph()
|
||||||
|
for signal_item in _signal_graph.signals:
|
||||||
|
_connect_to_signal(signal_item)
|
||||||
|
EngineDebugger.send_message(
|
||||||
|
"signal_debugger:generated_graph",
|
||||||
|
[[_signal_graph.signals.map(func (item): return item.dictionary_representation), _signal_graph.edges.map(func (item): return item.dictionary_representation)]]
|
||||||
|
)
|
||||||
|
if message == "stop" and _signal_graph:
|
||||||
|
for signal_item in _signal_graph.signals:
|
||||||
|
_disconnect_from_signal(signal_item)
|
||||||
|
|
||||||
|
if message == "invoke_signal" and data.size() == 2:
|
||||||
|
var node_name = data[0]
|
||||||
|
var signal_name = data[1]
|
||||||
|
|
||||||
|
var root_node = get_tree().current_scene
|
||||||
|
var node = root_node if root_node.name == node_name else root_node.find_child(node_name)
|
||||||
|
if node:
|
||||||
|
var connection_list = node.get_signal_connection_list(signal_name)
|
||||||
|
for connection in connection_list:
|
||||||
|
var callable = connection["callable"]
|
||||||
|
var bound_args = callable.get_bound_arguments()
|
||||||
|
var bound_args_count = callable.get_bound_arguments_count()
|
||||||
|
var method = callable.get_method()
|
||||||
|
callable.callv([node])
|
||||||
|
|
||||||
|
return true
|
||||||
|
|
||||||
|
func _on_signal_execution(signal_name: String, node_name: String, args):
|
||||||
|
EngineDebugger.send_message(
|
||||||
|
"signal_debugger:signal_executed",
|
||||||
|
[Time.get_datetime_string_from_system(), node_name, signal_name]
|
||||||
|
)
|
||||||
|
|
||||||
|
# Methods
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
|
||||||
|
func generate_signal_graph() -> SignalGraph:
|
||||||
|
var graph = SignalGraphUtility.create_signal_graph_from_node(get_tree().current_scene)
|
||||||
|
return graph
|
||||||
|
#var signal_graph = SignalGraph.new(get_tree().current_scene.name)
|
||||||
|
#var all_nodes: Array[Node] = _gather_nodes_in_scene()
|
||||||
|
#var signals: Array[SignalDescription] = []
|
||||||
|
#var edges: Array[SignalConnection] = []
|
||||||
|
#
|
||||||
|
#for node in all_nodes:
|
||||||
|
#for signal_item in node.get_signal_list():
|
||||||
|
#var existing_signals = []
|
||||||
|
#var connection_list = node.get_signal_connection_list(signal_item["name"] as String)
|
||||||
|
#if connection_list.size() > 0:
|
||||||
|
#for connection in connection_list:
|
||||||
|
#var should_display_connection = "name" in connection["callable"].get_object() and not connection["callable"].get_object().name.begins_with("@")
|
||||||
|
#if should_display_connection:
|
||||||
|
#var signal_description: SignalDescription
|
||||||
|
#var filtered_signals = existing_signals.filter(func (element): return element.signal_name == signal_item.name and element.node_name == node.name)
|
||||||
|
#if filtered_signals.size() == 1:
|
||||||
|
#signal_description = filtered_signals[0]
|
||||||
|
#else:
|
||||||
|
#signal_description = SignalDescription.new(node.name, signal_item.name)
|
||||||
|
#existing_signals.append(signal_description)
|
||||||
|
#signals.append(signal_description)
|
||||||
|
#
|
||||||
|
#var signal_edge = SignalConnection.new(signal_description.id, signal_description.node_name, connection["callable"].get_object().name, connection["callable"].get_method())
|
||||||
|
#if not signal_graph.edges.any(func (element): return element.signal_id == signal_description.id):
|
||||||
|
#edges.append(signal_edge)
|
||||||
|
#
|
||||||
|
#var temp_signals = {}
|
||||||
|
#for item in signals:
|
||||||
|
#temp_signals[item.id] = item
|
||||||
|
#
|
||||||
|
#var temp_edges = {}
|
||||||
|
#for item in edges:
|
||||||
|
#temp_edges[item.dictionary_key] = item
|
||||||
|
#
|
||||||
|
#signal_graph.signals.assign(temp_signals.keys().map(func (key): return temp_signals[key]))
|
||||||
|
#signal_graph.edges.assign(temp_edges.keys().map(func (key): return temp_edges[key]))
|
||||||
|
#
|
||||||
|
#return signal_graph
|
||||||
|
|
||||||
|
#func _gather_nodes_in_scene() -> Array[Node]:
|
||||||
|
#var scene_root = get_tree().current_scene
|
||||||
|
#var node_list: Array[Node] = [scene_root]
|
||||||
|
#return node_list + _gather_nodes_from_node(scene_root)
|
||||||
|
#
|
||||||
|
#func _gather_nodes_from_node(node: Node) -> Array[Node]:
|
||||||
|
#var nodes: Array[Node] = []
|
||||||
|
#for child in node.get_children(false):
|
||||||
|
#nodes.append(child)
|
||||||
|
#nodes += _gather_nodes_from_node(child)
|
||||||
|
#
|
||||||
|
#return nodes
|
||||||
|
|
||||||
|
func _connect_to_signal(signal_item: SignalDescription):
|
||||||
|
var root_node = get_tree().current_scene
|
||||||
|
var _execute: Callable = func (args = []): _on_signal_execution(signal_item.signal_name, signal_item.node_name, args)
|
||||||
|
if root_node.name == signal_item.node_name:
|
||||||
|
root_node.connect(signal_item.signal_name, _execute)
|
||||||
|
_lambda_map[signal_item] = _execute
|
||||||
|
else:
|
||||||
|
var child = root_node.find_child(signal_item.node_name)
|
||||||
|
if child:
|
||||||
|
child.connect(signal_item.signal_name, _execute)
|
||||||
|
_lambda_map[signal_item] = _execute
|
||||||
|
|
||||||
|
func _disconnect_from_signal(signal_item: SignalDescription):
|
||||||
|
var root_node = get_tree().current_scene
|
||||||
|
if root_node.name == signal_item.node_name:
|
||||||
|
var callable = _lambda_map[signal_item]
|
||||||
|
if callable:
|
||||||
|
root_node.disconnect(signal_item.signal_name, callable)
|
||||||
|
_lambda_map.erase(signal_item)
|
||||||
|
else:
|
||||||
|
var child = root_node.find_child(signal_item.node_name)
|
||||||
|
if child:
|
||||||
|
var callable = _lambda_map[signal_item]
|
||||||
|
if callable:
|
||||||
|
child.disconnect(signal_item.signal_name, callable)
|
||||||
|
_lambda_map.erase(signal_item)
|
97
addons/SignalVisualizer/Debugger/SignalDebugger.tscn
Normal file
|
@ -0,0 +1,97 @@
|
||||||
|
[gd_scene load_steps=5 format=3 uid="uid://cbsmvov8u78q"]
|
||||||
|
|
||||||
|
[ext_resource type="Script" path="res://addons/SignalVisualizer/Debugger/signal_debugger_panel.gd" id="1_66cpc"]
|
||||||
|
[ext_resource type="Texture2D" uid="uid://be3nwoioa311t" path="res://addons/SignalVisualizer/Play.svg" id="2_2wkuv"]
|
||||||
|
[ext_resource type="Texture2D" uid="uid://oo1oq2colx5b" path="res://addons/SignalVisualizer/Stop.svg" id="3_bg5eu"]
|
||||||
|
[ext_resource type="Texture2D" uid="uid://bmnff63evbdhv" path="res://addons/SignalVisualizer/Clear.svg" id="4_vg63r"]
|
||||||
|
|
||||||
|
[node name="SignalDebugger" type="Control"]
|
||||||
|
clip_contents = true
|
||||||
|
layout_mode = 3
|
||||||
|
anchors_preset = 15
|
||||||
|
anchor_right = 1.0
|
||||||
|
anchor_bottom = 1.0
|
||||||
|
grow_horizontal = 2
|
||||||
|
grow_vertical = 2
|
||||||
|
script = ExtResource("1_66cpc")
|
||||||
|
start_icon = ExtResource("2_2wkuv")
|
||||||
|
stop_icon = ExtResource("3_bg5eu")
|
||||||
|
|
||||||
|
[node name="VBoxContainer" type="VBoxContainer" parent="."]
|
||||||
|
clip_contents = true
|
||||||
|
layout_mode = 1
|
||||||
|
anchors_preset = 15
|
||||||
|
anchor_right = 1.0
|
||||||
|
anchor_bottom = 1.0
|
||||||
|
grow_horizontal = 2
|
||||||
|
grow_vertical = 2
|
||||||
|
|
||||||
|
[node name="HBoxContainer" type="HBoxContainer" parent="VBoxContainer"]
|
||||||
|
custom_minimum_size = Vector2(2.08165e-12, 50)
|
||||||
|
layout_mode = 2
|
||||||
|
theme_override_constants/separation = 8
|
||||||
|
|
||||||
|
[node name="ActionButton" type="Button" parent="VBoxContainer/HBoxContainer"]
|
||||||
|
unique_name_in_owner = true
|
||||||
|
layout_mode = 2
|
||||||
|
disabled = true
|
||||||
|
text = "Start"
|
||||||
|
icon = ExtResource("2_2wkuv")
|
||||||
|
|
||||||
|
[node name="ClearAllButton" type="Button" parent="VBoxContainer/HBoxContainer"]
|
||||||
|
unique_name_in_owner = true
|
||||||
|
layout_mode = 2
|
||||||
|
text = "Clear All"
|
||||||
|
icon = ExtResource("4_vg63r")
|
||||||
|
|
||||||
|
[node name="Spacer" type="Control" parent="VBoxContainer/HBoxContainer"]
|
||||||
|
layout_mode = 2
|
||||||
|
size_flags_horizontal = 3
|
||||||
|
|
||||||
|
[node name="ClearLogsButton" type="Button" parent="VBoxContainer/HBoxContainer"]
|
||||||
|
layout_mode = 2
|
||||||
|
text = "Clear Logs"
|
||||||
|
icon = ExtResource("4_vg63r")
|
||||||
|
|
||||||
|
[node name="HSplitContainer" type="HSplitContainer" parent="VBoxContainer"]
|
||||||
|
layout_mode = 2
|
||||||
|
size_flags_vertical = 3
|
||||||
|
|
||||||
|
[node name="SignalTree" type="Tree" parent="VBoxContainer/HSplitContainer"]
|
||||||
|
unique_name_in_owner = true
|
||||||
|
custom_minimum_size = Vector2(250, 2.08165e-12)
|
||||||
|
layout_mode = 2
|
||||||
|
columns = 2
|
||||||
|
allow_reselect = true
|
||||||
|
allow_rmb_select = true
|
||||||
|
hide_root = true
|
||||||
|
|
||||||
|
[node name="VBoxContainer" type="VBoxContainer" parent="VBoxContainer/HSplitContainer"]
|
||||||
|
layout_mode = 2
|
||||||
|
|
||||||
|
[node name="TabBar" type="TabBar" parent="VBoxContainer/HSplitContainer/VBoxContainer"]
|
||||||
|
layout_mode = 2
|
||||||
|
tab_count = 2
|
||||||
|
tab_0/title = "Signal Log"
|
||||||
|
tab_1/title = "Signal Graph"
|
||||||
|
|
||||||
|
[node name="LogLabel" type="RichTextLabel" parent="VBoxContainer/HSplitContainer/VBoxContainer"]
|
||||||
|
unique_name_in_owner = true
|
||||||
|
layout_mode = 2
|
||||||
|
size_flags_horizontal = 3
|
||||||
|
size_flags_vertical = 3
|
||||||
|
theme_override_colors/default_color = Color(0.690196, 0.690196, 0.690196, 1)
|
||||||
|
bbcode_enabled = true
|
||||||
|
scroll_following = true
|
||||||
|
|
||||||
|
[node name="Graph" type="GraphEdit" parent="VBoxContainer/HSplitContainer/VBoxContainer"]
|
||||||
|
unique_name_in_owner = true
|
||||||
|
visible = false
|
||||||
|
layout_mode = 2
|
||||||
|
size_flags_vertical = 3
|
||||||
|
|
||||||
|
[connection signal="pressed" from="VBoxContainer/HBoxContainer/ActionButton" to="." method="_on_action_button_pressed"]
|
||||||
|
[connection signal="pressed" from="VBoxContainer/HBoxContainer/ClearAllButton" to="." method="_on_clear_all_button_pressed"]
|
||||||
|
[connection signal="pressed" from="VBoxContainer/HBoxContainer/ClearLogsButton" to="." method="_on_clear_logs_button_pressed"]
|
||||||
|
[connection signal="item_selected" from="VBoxContainer/HSplitContainer/SignalTree" to="." method="_on_signal_tree_item_selected"]
|
||||||
|
[connection signal="tab_changed" from="VBoxContainer/HSplitContainer/VBoxContainer/TabBar" to="." method="_on_tab_bar_tab_changed"]
|
192
addons/SignalVisualizer/Debugger/signal_debugger_panel.gd
Normal file
|
@ -0,0 +1,192 @@
|
||||||
|
@tool
|
||||||
|
class_name SignalDebuggerPanel extends Control
|
||||||
|
|
||||||
|
signal open_script(node_name: String, method_signature: String)
|
||||||
|
|
||||||
|
signal start_signal_debugging
|
||||||
|
signal stop_signal_debugging
|
||||||
|
|
||||||
|
var SignalGraphNode = preload("res://addons/SignalVisualizer/Visualizer/signal_graph_node.tscn")
|
||||||
|
var GraphNodeItem = preload("res://addons/SignalVisualizer/Visualizer/signal_graph_node_item.tscn")
|
||||||
|
|
||||||
|
const SOURCE_COLOR: Color = Color.SKY_BLUE
|
||||||
|
const DESTINATION_COLOR: Color = Color.CORAL
|
||||||
|
const CONNECTION_TYPE: int = 0
|
||||||
|
|
||||||
|
enum Tabs {
|
||||||
|
LOG,
|
||||||
|
GRAPH
|
||||||
|
}
|
||||||
|
|
||||||
|
# Properties
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
|
||||||
|
@export var start_icon: Texture2D
|
||||||
|
@export var stop_icon: Texture2D
|
||||||
|
|
||||||
|
@onready var action_button: Button = %ActionButton
|
||||||
|
@onready var clear_all_button: Button = %ClearAllButton
|
||||||
|
@onready var signal_tree: Tree = %SignalTree
|
||||||
|
@onready var log_label: RichTextLabel = %LogLabel
|
||||||
|
@onready var graph_node: GraphEdit = %Graph
|
||||||
|
|
||||||
|
var is_started: bool = false :
|
||||||
|
get: return is_started
|
||||||
|
set(new_value):
|
||||||
|
is_started = new_value
|
||||||
|
_update_action_button()
|
||||||
|
|
||||||
|
var _signals: Array = []
|
||||||
|
var _signal_filter: Array = []
|
||||||
|
var _is_stack_trace_enabled: bool = false
|
||||||
|
var _debugger_tab_state: Tabs = Tabs.LOG
|
||||||
|
var _graph: SignalGraph
|
||||||
|
|
||||||
|
# Lifecycle
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
|
||||||
|
func _ready():
|
||||||
|
disable()
|
||||||
|
_handle_tab_update(0)
|
||||||
|
|
||||||
|
# Signals
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
|
||||||
|
func _on_action_button_pressed():
|
||||||
|
if is_started:
|
||||||
|
stop()
|
||||||
|
else:
|
||||||
|
start()
|
||||||
|
|
||||||
|
func _on_clear_all_button_pressed():
|
||||||
|
log_label.clear()
|
||||||
|
signal_tree.clear()
|
||||||
|
graph_node.clear_connections()
|
||||||
|
for child in graph_node.get_children():
|
||||||
|
if child is SignalGraphNode:
|
||||||
|
child.queue_free()
|
||||||
|
|
||||||
|
func _on_clear_logs_button_pressed():
|
||||||
|
log_label.clear()
|
||||||
|
|
||||||
|
func _on_signal_tree_item_selected():
|
||||||
|
# Updates the checkmark button
|
||||||
|
var selected_item = signal_tree.get_selected()
|
||||||
|
var is_checked = selected_item.is_checked(1)
|
||||||
|
selected_item.set_checked(1, (not is_checked))
|
||||||
|
|
||||||
|
# Add / Remove signal from filters
|
||||||
|
var selected_signal = _signals.filter(func (element): return element.signal_name == selected_item.get_text(0))[0]
|
||||||
|
if _signal_filter.has(selected_signal.signal_name):
|
||||||
|
var selected_index = _signal_filter.find(selected_signal.signal_name)
|
||||||
|
_signal_filter.remove_at(selected_index)
|
||||||
|
else:
|
||||||
|
_signal_filter.append(selected_signal.signal_name)
|
||||||
|
|
||||||
|
func _on_tab_bar_tab_changed(tab: int):
|
||||||
|
_handle_tab_update(tab)
|
||||||
|
|
||||||
|
func _on_stack_trace_button_pressed():
|
||||||
|
_is_stack_trace_enabled = not _is_stack_trace_enabled
|
||||||
|
|
||||||
|
func _on_open_signal_in_script(data: SignalGraphNodeItem.Metadata):
|
||||||
|
open_script.emit(data.node_name, data.method_signature)
|
||||||
|
|
||||||
|
# Methods
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
|
||||||
|
func enable():
|
||||||
|
action_button.disabled = false
|
||||||
|
|
||||||
|
func disable():
|
||||||
|
action_button.disabled = true
|
||||||
|
|
||||||
|
func start():
|
||||||
|
if not is_started:
|
||||||
|
is_started = true
|
||||||
|
action_button.icon = stop_icon
|
||||||
|
start_signal_debugging.emit()
|
||||||
|
log_label.append_text("[color=#B0B0B0]Signal Debugging Started...[/color]")
|
||||||
|
log_label.newline()
|
||||||
|
log_label.newline()
|
||||||
|
|
||||||
|
func stop():
|
||||||
|
if is_started:
|
||||||
|
is_started = false
|
||||||
|
action_button.icon = start_icon
|
||||||
|
stop_signal_debugging.emit()
|
||||||
|
log_label.newline()
|
||||||
|
log_label.append_text("[color=#B0B0B0]Signal Debugging Stopped[/color]")
|
||||||
|
log_label.newline()
|
||||||
|
log_label.newline()
|
||||||
|
|
||||||
|
func create_tree_from_signals(signals: Array):
|
||||||
|
_signals = signals
|
||||||
|
var root = signal_tree.create_item()
|
||||||
|
root.set_text(0, "Signals")
|
||||||
|
|
||||||
|
var tree_items: Dictionary = {}
|
||||||
|
|
||||||
|
for signal_item in signals:
|
||||||
|
var node_tree_item: TreeItem
|
||||||
|
if tree_items.has(signal_item.node_name):
|
||||||
|
node_tree_item = tree_items[signal_item.node_name] as TreeItem
|
||||||
|
else:
|
||||||
|
node_tree_item = signal_tree.create_item(root)
|
||||||
|
node_tree_item.set_text(0, signal_item.node_name)
|
||||||
|
node_tree_item.set_selectable(0, false)
|
||||||
|
node_tree_item.set_selectable(1, false)
|
||||||
|
tree_items[signal_item.node_name] = node_tree_item
|
||||||
|
|
||||||
|
var signal_tree_item = signal_tree.create_item(node_tree_item)
|
||||||
|
signal_tree_item.set_text(0, signal_item.signal_name)
|
||||||
|
signal_tree_item.set_cell_mode(1, TreeItem.CELL_MODE_CHECK)
|
||||||
|
signal_tree_item.set_checked(1, true)
|
||||||
|
signal_tree_item.set_selectable(0, false)
|
||||||
|
signal_tree_item.set_selectable(1, true)
|
||||||
|
|
||||||
|
func create_signal_graph(signals: Array, edges: Array):
|
||||||
|
_graph = SignalGraphUtility.create_signal_graph(get_tree().edited_scene_root.scene_file_path, signals, edges)
|
||||||
|
SignalGraphUtility.generate_signal_graph_nodes(_graph, graph_node, _on_open_signal_in_script)
|
||||||
|
|
||||||
|
func log_signal_execution(time: String, node_name: String, signal_name: String):
|
||||||
|
if _signal_filter != null and _signal_filter.has(signal_name):
|
||||||
|
return
|
||||||
|
|
||||||
|
if not log_label.text.is_empty():
|
||||||
|
log_label.newline()
|
||||||
|
log_label.append_text(
|
||||||
|
"[color=#FFCC00]{time}[/color]\t\t{node_name}\t\t{signal_name}".format({ "time": time, "node_name": node_name, "signal_name": signal_name })
|
||||||
|
)
|
||||||
|
log_label.newline()
|
||||||
|
|
||||||
|
func _handle_tab_update(selected_tab_index: int):
|
||||||
|
match selected_tab_index:
|
||||||
|
1:
|
||||||
|
_debugger_tab_state = Tabs.GRAPH
|
||||||
|
_:
|
||||||
|
_debugger_tab_state = Tabs.LOG
|
||||||
|
|
||||||
|
match _debugger_tab_state:
|
||||||
|
Tabs.LOG:
|
||||||
|
log_label.show()
|
||||||
|
graph_node.hide()
|
||||||
|
Tabs.GRAPH:
|
||||||
|
log_label.hide()
|
||||||
|
graph_node.show()
|
||||||
|
|
||||||
|
func _update_action_button():
|
||||||
|
if is_started:
|
||||||
|
action_button.text = "Stop"
|
||||||
|
action_button.modulate = Color("#ff3b30")
|
||||||
|
else:
|
||||||
|
action_button.text = "Start"
|
||||||
|
action_button.modulate = Color.WHITE
|
1
addons/SignalVisualizer/GraphEdit.svg
Normal file
|
@ -0,0 +1 @@
|
||||||
|
<svg height="24" width="24" viewBox="0 0 16 16" xmlns="http://www.w3.org/2000/svg"><path d="M11 1a1 1 0 0 0-1 1v3a1 1 0 0 0 1 1h3a1 1 0 0 0 1-1V2a1 1 0 0 0-1-1zM6.732 5A2 2 0 0 1 7 6v1.117L9.268 6A2 2 0 0 1 9 5V3.883zM2 5a1 1 0 0 0-1 1v4a1 1 0 0 0 1 1h3a1 1 0 0 0 1-1V6a1 1 0 0 0-1-1zm5 3.883V10a2 2 0 0 1-.268 1L9 12.117V11a2 2 0 0 1 .268-1zM11 10a1 1 0 0 0-1 1v3a1 1 0 0 0 1 1h3a1 1 0 0 0 1-1v-3a1 1 0 0 0-1-1z" fill="#8eef97"/></svg>
|
After Width: | Height: | Size: 437 B |
37
addons/SignalVisualizer/GraphEdit.svg.import
Normal file
|
@ -0,0 +1,37 @@
|
||||||
|
[remap]
|
||||||
|
|
||||||
|
importer="texture"
|
||||||
|
type="CompressedTexture2D"
|
||||||
|
uid="uid://bxj8ep08wbnm6"
|
||||||
|
path="res://.godot/imported/GraphEdit.svg-90dae61e8e0b157ab8eff95fe4b91e53.ctex"
|
||||||
|
metadata={
|
||||||
|
"vram_texture": false
|
||||||
|
}
|
||||||
|
|
||||||
|
[deps]
|
||||||
|
|
||||||
|
source_file="res://addons/SignalVisualizer/GraphEdit.svg"
|
||||||
|
dest_files=["res://.godot/imported/GraphEdit.svg-90dae61e8e0b157ab8eff95fe4b91e53.ctex"]
|
||||||
|
|
||||||
|
[params]
|
||||||
|
|
||||||
|
compress/mode=0
|
||||||
|
compress/high_quality=false
|
||||||
|
compress/lossy_quality=0.7
|
||||||
|
compress/hdr_compression=1
|
||||||
|
compress/normal_map=0
|
||||||
|
compress/channel_pack=0
|
||||||
|
mipmaps/generate=false
|
||||||
|
mipmaps/limit=-1
|
||||||
|
roughness/mode=0
|
||||||
|
roughness/src_normal=""
|
||||||
|
process/fix_alpha_border=true
|
||||||
|
process/premult_alpha=false
|
||||||
|
process/normal_map_invert_y=false
|
||||||
|
process/hdr_as_srgb=false
|
||||||
|
process/hdr_clamp_exposure=false
|
||||||
|
process/size_limit=0
|
||||||
|
detect_3d/compress_to=1
|
||||||
|
svg/scale=1.0
|
||||||
|
editor/scale_with_editor_scale=false
|
||||||
|
editor/convert_colors_with_editor_theme=false
|
1
addons/SignalVisualizer/Play.svg
Normal file
|
@ -0,0 +1 @@
|
||||||
|
<svg height="24" viewBox="0 0 16 16" width="24" xmlns="http://www.w3.org/2000/svg"><path d="M4 12a1 1 0 0 0 1.555.832l6-4a1 1 0 0 0 0-1.664l-6-4A1 1 0 0 0 4 4z" fill="#e0e0e0"/></svg>
|
After Width: | Height: | Size: 184 B |
37
addons/SignalVisualizer/Play.svg.import
Normal file
|
@ -0,0 +1,37 @@
|
||||||
|
[remap]
|
||||||
|
|
||||||
|
importer="texture"
|
||||||
|
type="CompressedTexture2D"
|
||||||
|
uid="uid://be3nwoioa311t"
|
||||||
|
path="res://.godot/imported/Play.svg-a446691ffcef211028bb160b5a2d6ff1.ctex"
|
||||||
|
metadata={
|
||||||
|
"vram_texture": false
|
||||||
|
}
|
||||||
|
|
||||||
|
[deps]
|
||||||
|
|
||||||
|
source_file="res://addons/SignalVisualizer/Play.svg"
|
||||||
|
dest_files=["res://.godot/imported/Play.svg-a446691ffcef211028bb160b5a2d6ff1.ctex"]
|
||||||
|
|
||||||
|
[params]
|
||||||
|
|
||||||
|
compress/mode=0
|
||||||
|
compress/high_quality=false
|
||||||
|
compress/lossy_quality=0.7
|
||||||
|
compress/hdr_compression=1
|
||||||
|
compress/normal_map=0
|
||||||
|
compress/channel_pack=0
|
||||||
|
mipmaps/generate=false
|
||||||
|
mipmaps/limit=-1
|
||||||
|
roughness/mode=0
|
||||||
|
roughness/src_normal=""
|
||||||
|
process/fix_alpha_border=true
|
||||||
|
process/premult_alpha=false
|
||||||
|
process/normal_map_invert_y=false
|
||||||
|
process/hdr_as_srgb=false
|
||||||
|
process/hdr_clamp_exposure=false
|
||||||
|
process/size_limit=0
|
||||||
|
detect_3d/compress_to=1
|
||||||
|
svg/scale=1.0
|
||||||
|
editor/scale_with_editor_scale=false
|
||||||
|
editor/convert_colors_with_editor_theme=false
|
162
addons/SignalVisualizer/SignalVisualizer.gd
Normal file
|
@ -0,0 +1,162 @@
|
||||||
|
@tool
|
||||||
|
extends EditorPlugin
|
||||||
|
|
||||||
|
class SignalDebuggerPlugin extends EditorDebuggerPlugin:
|
||||||
|
var SignalDebuggerPanelScene = preload("res://addons/SignalVisualizer/Debugger/SignalDebugger.tscn")
|
||||||
|
|
||||||
|
signal open_script
|
||||||
|
signal start_signal_debugging
|
||||||
|
signal stop_signal_debugging
|
||||||
|
|
||||||
|
var debugger_panel
|
||||||
|
|
||||||
|
func _has_capture(prefix) -> bool:
|
||||||
|
return prefix == "signal_debugger"
|
||||||
|
|
||||||
|
func _capture(message, data, session_id) -> bool:
|
||||||
|
if message == "signal_debugger:signal_executed":
|
||||||
|
if data.size() == 3:
|
||||||
|
var time = data[0]
|
||||||
|
var node_name = data[1]
|
||||||
|
var signal_name = data[2]
|
||||||
|
debugger_panel.log_signal_execution(time, node_name, signal_name)
|
||||||
|
return true
|
||||||
|
|
||||||
|
if message == "signal_debugger:generated_graph":
|
||||||
|
if data.size() == 1:
|
||||||
|
var signals = data[0][0] as Array
|
||||||
|
var edges = data[0][1] as Array
|
||||||
|
debugger_panel.create_tree_from_signals(signals)
|
||||||
|
debugger_panel.create_signal_graph(signals, edges)
|
||||||
|
return true
|
||||||
|
|
||||||
|
return false
|
||||||
|
|
||||||
|
func _setup_session(session_id):
|
||||||
|
debugger_panel = SignalDebuggerPanelScene.instantiate()
|
||||||
|
var session = get_session(session_id)
|
||||||
|
|
||||||
|
debugger_panel.name = "Signal Debugger"
|
||||||
|
debugger_panel.open_script.connect(func (arg1, arg2): open_script.emit(arg1, arg2))
|
||||||
|
debugger_panel.start_signal_debugging.connect(func (): start_signal_debugging.emit())
|
||||||
|
debugger_panel.stop_signal_debugging.connect(func (): stop_signal_debugging.emit())
|
||||||
|
|
||||||
|
session.started.connect(
|
||||||
|
func ():
|
||||||
|
debugger_panel.enable()
|
||||||
|
)
|
||||||
|
session.stopped.connect(
|
||||||
|
func ():
|
||||||
|
debugger_panel.stop()
|
||||||
|
debugger_panel.disable()
|
||||||
|
stop_signal_debugging.emit()
|
||||||
|
)
|
||||||
|
|
||||||
|
session.add_session_tab(debugger_panel)
|
||||||
|
|
||||||
|
var SignalVisualizerDockScene = preload("res://addons/SignalVisualizer/Visualizer/signal_visualizer_dock.tscn")
|
||||||
|
|
||||||
|
class ScriptMethodReference:
|
||||||
|
var script_reference: Script
|
||||||
|
var line_number: int
|
||||||
|
|
||||||
|
# Properties
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
|
||||||
|
var dock: Control
|
||||||
|
var debugger: SignalDebuggerPlugin
|
||||||
|
|
||||||
|
# Lifecycle
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
|
||||||
|
func _enter_tree():
|
||||||
|
dock = SignalVisualizerDockScene.instantiate()
|
||||||
|
debugger = SignalDebuggerPlugin.new()
|
||||||
|
|
||||||
|
dock.open_script.connect(_on_open_signal_in_script)
|
||||||
|
add_control_to_bottom_panel(dock, "Signal Visualizer")
|
||||||
|
|
||||||
|
debugger.start_signal_debugging.connect(_on_debugger_start_signal_debugging)
|
||||||
|
debugger.stop_signal_debugging.connect(_on_debugger_stop_signal_debugging)
|
||||||
|
debugger.open_script.connect(_on_open_signal_in_script)
|
||||||
|
add_debugger_plugin(debugger)
|
||||||
|
|
||||||
|
if not ProjectSettings.has_setting("autoload/Signal_Debugger"):
|
||||||
|
add_autoload_singleton("Signal_Debugger", "res://addons/SignalVisualizer/Debugger/SignalDebugger.gd")
|
||||||
|
|
||||||
|
func _exit_tree():
|
||||||
|
remove_control_from_bottom_panel(dock)
|
||||||
|
dock.free()
|
||||||
|
|
||||||
|
remove_debugger_plugin(debugger)
|
||||||
|
remove_autoload_singleton("Signal_Debugger")
|
||||||
|
|
||||||
|
# Signals
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
|
||||||
|
func _on_open_signal_in_script(node_name: String, method_signature: String):
|
||||||
|
var node: Node
|
||||||
|
if get_tree().edited_scene_root.name == node_name:
|
||||||
|
node = get_tree().edited_scene_root
|
||||||
|
else:
|
||||||
|
node = get_tree().edited_scene_root.find_child(node_name)
|
||||||
|
|
||||||
|
if node != null:
|
||||||
|
var script: Script = node.get_script()
|
||||||
|
if script != null:
|
||||||
|
var editor = get_editor_interface()
|
||||||
|
var method_reference = _find_method_reference_in_script(script, method_signature)
|
||||||
|
|
||||||
|
if method_reference != null:
|
||||||
|
editor.edit_script(method_reference.script_reference, method_reference.line_number, 0)
|
||||||
|
editor.set_main_screen_editor("Script")
|
||||||
|
else:
|
||||||
|
push_warning("Requested method in script ({script}) for node ({name}) is not available.".format({ "name": node_name, "script": script.name }))
|
||||||
|
else:
|
||||||
|
push_warning("Requested script for node ({name}) is not available.".format({ "name": node_name }))
|
||||||
|
else:
|
||||||
|
push_warning("Requested script for node ({name}) is not available.".format({ "name": node_name }))
|
||||||
|
|
||||||
|
func _on_debugger_start_signal_debugging():
|
||||||
|
for session in debugger.get_sessions():
|
||||||
|
session.send_message("signal_debugger:start", [])
|
||||||
|
|
||||||
|
func _on_debugger_stop_signal_debugging():
|
||||||
|
for session in debugger.get_sessions():
|
||||||
|
session.send_message("signal_debugger:stop", [])
|
||||||
|
|
||||||
|
# Methods
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
|
||||||
|
func _find_method_reference_in_script(script: Script, method_signature: String) -> ScriptMethodReference:
|
||||||
|
var line_number = __find_method_line_number_in_script(script, method_signature)
|
||||||
|
|
||||||
|
if line_number == -1:
|
||||||
|
var base_script = script.get_base_script()
|
||||||
|
if base_script:
|
||||||
|
return _find_method_reference_in_script(base_script, method_signature)
|
||||||
|
|
||||||
|
var reference = ScriptMethodReference.new()
|
||||||
|
reference.script_reference = script
|
||||||
|
reference.line_number = line_number
|
||||||
|
|
||||||
|
return reference
|
||||||
|
|
||||||
|
func __find_method_line_number_in_script(script: Script, method_signature: String) -> int:
|
||||||
|
var line_number = 0
|
||||||
|
var found = false
|
||||||
|
for line in script.source_code.split("\n", true):
|
||||||
|
line_number += 1
|
||||||
|
if line.contains(method_signature):
|
||||||
|
found = true
|
||||||
|
return line_number
|
||||||
|
|
||||||
|
return -1
|
1
addons/SignalVisualizer/Stop.svg
Normal file
|
@ -0,0 +1 @@
|
||||||
|
<svg height="24" viewBox="0 0 16 16" width="24" xmlns="http://www.w3.org/2000/svg"><rect x="3" y="3" height="10" width="10" rx="1" fill="#e0e0e0"/></svg>
|
After Width: | Height: | Size: 154 B |
37
addons/SignalVisualizer/Stop.svg.import
Normal file
|
@ -0,0 +1,37 @@
|
||||||
|
[remap]
|
||||||
|
|
||||||
|
importer="texture"
|
||||||
|
type="CompressedTexture2D"
|
||||||
|
uid="uid://oo1oq2colx5b"
|
||||||
|
path="res://.godot/imported/Stop.svg-e085086fb31c334bc2f02ca2bffba522.ctex"
|
||||||
|
metadata={
|
||||||
|
"vram_texture": false
|
||||||
|
}
|
||||||
|
|
||||||
|
[deps]
|
||||||
|
|
||||||
|
source_file="res://addons/SignalVisualizer/Stop.svg"
|
||||||
|
dest_files=["res://.godot/imported/Stop.svg-e085086fb31c334bc2f02ca2bffba522.ctex"]
|
||||||
|
|
||||||
|
[params]
|
||||||
|
|
||||||
|
compress/mode=0
|
||||||
|
compress/high_quality=false
|
||||||
|
compress/lossy_quality=0.7
|
||||||
|
compress/hdr_compression=1
|
||||||
|
compress/normal_map=0
|
||||||
|
compress/channel_pack=0
|
||||||
|
mipmaps/generate=false
|
||||||
|
mipmaps/limit=-1
|
||||||
|
roughness/mode=0
|
||||||
|
roughness/src_normal=""
|
||||||
|
process/fix_alpha_border=true
|
||||||
|
process/premult_alpha=false
|
||||||
|
process/normal_map_invert_y=false
|
||||||
|
process/hdr_as_srgb=false
|
||||||
|
process/hdr_clamp_exposure=false
|
||||||
|
process/size_limit=0
|
||||||
|
detect_3d/compress_to=1
|
||||||
|
svg/scale=1.0
|
||||||
|
editor/scale_with_editor_scale=false
|
||||||
|
editor/convert_colors_with_editor_theme=false
|
31
addons/SignalVisualizer/Visualizer/resizable_label.gd
Normal file
|
@ -0,0 +1,31 @@
|
||||||
|
@tool
|
||||||
|
extends Label
|
||||||
|
|
||||||
|
# Properties
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
# Lifecycle
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
# Signals
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
# Methods
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
|
||||||
|
func get_text_size() -> Vector2:
|
||||||
|
return get_theme_default_font().get_string_size(text)
|
94
addons/SignalVisualizer/Visualizer/signal_graph_node.gd
Normal file
|
@ -0,0 +1,94 @@
|
||||||
|
@tool
|
||||||
|
class_name SignalGraphNode extends GraphNode
|
||||||
|
|
||||||
|
# Properties
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
|
||||||
|
var connections: Array = [] :
|
||||||
|
get: return connections
|
||||||
|
set(new_value):
|
||||||
|
connections = new_value
|
||||||
|
|
||||||
|
# Lifecycle
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
|
||||||
|
func _ready():
|
||||||
|
selectable = true
|
||||||
|
resizable = true
|
||||||
|
draggable = true
|
||||||
|
|
||||||
|
# Signals
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
|
||||||
|
func _on_resize_request(new_minsize):
|
||||||
|
size = new_minsize
|
||||||
|
|
||||||
|
# Methods
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
|
||||||
|
func has_source_signal_description(signal_name: String, destination_node_name: String) -> bool:
|
||||||
|
for child in get_children():
|
||||||
|
if child.name == "source_" + signal_name + "_" + destination_node_name:
|
||||||
|
return true
|
||||||
|
|
||||||
|
return false
|
||||||
|
|
||||||
|
func get_source_slot(signal_name: String, destination_node_name: String) -> int:
|
||||||
|
var index = 0
|
||||||
|
for child in get_children():
|
||||||
|
if child.name == "source_" + signal_name + "_" + destination_node_name:
|
||||||
|
return index
|
||||||
|
|
||||||
|
index += 1
|
||||||
|
|
||||||
|
return -1
|
||||||
|
|
||||||
|
func get_next_source_slot(signal_name: String, destination_node_name: String) -> int:
|
||||||
|
var index = 0
|
||||||
|
for child in get_children():
|
||||||
|
if child.name.begins_with("source_"):
|
||||||
|
if child.name == "source_" + signal_name + "_" + destination_node_name:
|
||||||
|
return index
|
||||||
|
|
||||||
|
index += 1
|
||||||
|
|
||||||
|
return -1
|
||||||
|
|
||||||
|
func has_destination_signal_description(signal_name: String, method_signature: String) -> bool:
|
||||||
|
for child in get_children():
|
||||||
|
if child.name == "destination_" + signal_name + "_" + _sanitize_method_signature(method_signature):
|
||||||
|
return true
|
||||||
|
|
||||||
|
return false
|
||||||
|
|
||||||
|
func get_destination_slot(signal_name: String, method_signature: String) -> int:
|
||||||
|
var index = 0
|
||||||
|
for child in get_children():
|
||||||
|
if child.name == "destination_" + signal_name + "_" + _sanitize_method_signature(method_signature):
|
||||||
|
return index
|
||||||
|
|
||||||
|
index += 1
|
||||||
|
|
||||||
|
return -1
|
||||||
|
|
||||||
|
func get_next_destination_slot(signal_name: String, method_signature: String) -> int:
|
||||||
|
var index = 0
|
||||||
|
for child in get_children():
|
||||||
|
if child.name.begins_with("destination_"):
|
||||||
|
if child.name == "destination_" + signal_name + "_" + _sanitize_method_signature(method_signature):
|
||||||
|
return index
|
||||||
|
|
||||||
|
index += 1
|
||||||
|
|
||||||
|
return -1
|
||||||
|
|
||||||
|
func _sanitize_method_signature(signature: String) -> String:
|
||||||
|
return signature.replace("::", "__")
|
12
addons/SignalVisualizer/Visualizer/signal_graph_node.tscn
Normal file
|
@ -0,0 +1,12 @@
|
||||||
|
[gd_scene load_steps=2 format=3 uid="uid://cq10iaub18e54"]
|
||||||
|
|
||||||
|
[ext_resource type="Script" path="res://addons/SignalVisualizer/Visualizer/signal_graph_node.gd" id="1_ovklj"]
|
||||||
|
|
||||||
|
[node name="SignalGraphNode" type="GraphNode"]
|
||||||
|
custom_minimum_size = Vector2(100, 50)
|
||||||
|
offset_right = 232.0
|
||||||
|
offset_bottom = 54.0
|
||||||
|
resizable = true
|
||||||
|
script = ExtResource("1_ovklj")
|
||||||
|
|
||||||
|
[connection signal="resize_request" from="." to="." method="_on_resize_request"]
|
57
addons/SignalVisualizer/Visualizer/signal_graph_node_item.gd
Normal file
|
@ -0,0 +1,57 @@
|
||||||
|
@tool
|
||||||
|
class_name SignalGraphNodeItem extends Control
|
||||||
|
|
||||||
|
signal open_script(metadata: SignalGraphNodeItem.Metadata)
|
||||||
|
|
||||||
|
class Metadata:
|
||||||
|
var signal_name: String
|
||||||
|
var method_signature: String
|
||||||
|
var node_name: String
|
||||||
|
|
||||||
|
func _init(signal_name: String, method_signature: String, node_name: String):
|
||||||
|
self.signal_name = signal_name
|
||||||
|
self.method_signature = method_signature
|
||||||
|
self.node_name = node_name
|
||||||
|
|
||||||
|
# Properties
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
|
||||||
|
@onready var label: Label = %DescriptionLabel
|
||||||
|
|
||||||
|
var signal_data: Metadata = null
|
||||||
|
|
||||||
|
var text: String = "" :
|
||||||
|
get: return text
|
||||||
|
set(new_value):
|
||||||
|
text = new_value
|
||||||
|
if label:
|
||||||
|
label.text = text
|
||||||
|
|
||||||
|
# Lifecycle
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
|
||||||
|
func _ready():
|
||||||
|
_update()
|
||||||
|
|
||||||
|
# Signals
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
|
||||||
|
func _on_open_signal_in_script_button_pressed():
|
||||||
|
open_script.emit(signal_data)
|
||||||
|
|
||||||
|
# Methods
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
|
||||||
|
func _update():
|
||||||
|
label.text = text
|
||||||
|
|
||||||
|
var text_size = label.get_text_size()
|
||||||
|
custom_minimum_size = Vector2((text_size.x * 2) + 50, text_size.y * 3)
|
|
@ -0,0 +1,43 @@
|
||||||
|
[gd_scene load_steps=3 format=3 uid="uid://b2lwtwp6kpwtb"]
|
||||||
|
|
||||||
|
[ext_resource type="Script" path="res://addons/SignalVisualizer/Visualizer/signal_graph_node_item.gd" id="1_jrd34"]
|
||||||
|
[ext_resource type="Script" path="res://addons/SignalVisualizer/Visualizer/resizable_label.gd" id="2_4wwd5"]
|
||||||
|
|
||||||
|
[node name="SignalItem" type="Control"]
|
||||||
|
clip_contents = true
|
||||||
|
custom_minimum_size = Vector2(51, 23)
|
||||||
|
layout_mode = 3
|
||||||
|
anchors_preset = 15
|
||||||
|
anchor_right = 1.0
|
||||||
|
anchor_bottom = 1.0
|
||||||
|
grow_horizontal = 2
|
||||||
|
grow_vertical = 2
|
||||||
|
size_flags_horizontal = 3
|
||||||
|
size_flags_vertical = 3
|
||||||
|
script = ExtResource("1_jrd34")
|
||||||
|
|
||||||
|
[node name="HBoxContainer" type="HBoxContainer" parent="."]
|
||||||
|
custom_minimum_size = Vector2(100, 50)
|
||||||
|
layout_mode = 1
|
||||||
|
anchors_preset = 15
|
||||||
|
anchor_right = 1.0
|
||||||
|
anchor_bottom = 1.0
|
||||||
|
grow_horizontal = 2
|
||||||
|
grow_vertical = 2
|
||||||
|
|
||||||
|
[node name="DescriptionLabel" type="Label" parent="HBoxContainer"]
|
||||||
|
unique_name_in_owner = true
|
||||||
|
custom_minimum_size = Vector2(100, 50)
|
||||||
|
layout_mode = 2
|
||||||
|
size_flags_horizontal = 3
|
||||||
|
size_flags_vertical = 1
|
||||||
|
vertical_alignment = 1
|
||||||
|
clip_text = true
|
||||||
|
script = ExtResource("2_4wwd5")
|
||||||
|
|
||||||
|
[node name="OpenSignalInScriptButton" type="Button" parent="HBoxContainer"]
|
||||||
|
layout_mode = 2
|
||||||
|
text = "Open"
|
||||||
|
flat = true
|
||||||
|
|
||||||
|
[connection signal="pressed" from="HBoxContainer/OpenSignalInScriptButton" to="." method="_on_open_signal_in_script_button_pressed"]
|
67
addons/SignalVisualizer/Visualizer/signal_visualizer_dock.gd
Normal file
|
@ -0,0 +1,67 @@
|
||||||
|
@tool
|
||||||
|
extends Control
|
||||||
|
|
||||||
|
signal open_script(node_name: String, method_signature: String)
|
||||||
|
|
||||||
|
var SignalGraphNode = preload("res://addons/SignalVisualizer/Visualizer/signal_graph_node.tscn")
|
||||||
|
var GraphNodeItem = preload("res://addons/SignalVisualizer/Visualizer/signal_graph_node_item.tscn")
|
||||||
|
|
||||||
|
# Properties
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
|
||||||
|
const SOURCE_COLOR: Color = Color.SKY_BLUE
|
||||||
|
const DESTINATION_COLOR: Color = Color.CORAL
|
||||||
|
const CONNECTION_TYPE: int = 0
|
||||||
|
|
||||||
|
@onready var arrange_nodes_checkbox: CheckBox = %ArrangeNodesCheckBox
|
||||||
|
@onready var signal_details_checkbox: CheckBox = %SignalDetailsCheckBox
|
||||||
|
@onready var signal_tree: Tree = %SignalTree
|
||||||
|
@onready var graph: GraphEdit = %Graph
|
||||||
|
|
||||||
|
# Lifecycle
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
# Signals
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
|
||||||
|
func _on_clear_graph_button_pressed():
|
||||||
|
clear()
|
||||||
|
|
||||||
|
func _on_generate_graph_button_pressed():
|
||||||
|
clear()
|
||||||
|
|
||||||
|
var scene_signal_graph = SignalGraphUtility.create_signal_graph_from_node(get_tree().edited_scene_root, true)
|
||||||
|
SignalGraphUtility.generate_signal_graph_nodes(scene_signal_graph, graph, _on_open_signal_in_script)
|
||||||
|
SignalGraphUtility.generate_signal_graph_tree(scene_signal_graph, signal_tree)
|
||||||
|
|
||||||
|
if arrange_nodes_checkbox.button_pressed:
|
||||||
|
graph.arrange_nodes()
|
||||||
|
|
||||||
|
func _on_open_signal_in_script(data: SignalGraphNodeItem.Metadata):
|
||||||
|
open_script.emit(data.node_name, data.method_signature)
|
||||||
|
|
||||||
|
# Methods
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
# |===================================|
|
||||||
|
|
||||||
|
func clear():
|
||||||
|
_clear_graph_nodes()
|
||||||
|
_clear_tree()
|
||||||
|
|
||||||
|
func _clear_graph_nodes():
|
||||||
|
graph.clear_connections()
|
||||||
|
for child in graph.get_children():
|
||||||
|
if child is SignalGraphNode:
|
||||||
|
child.queue_free()
|
||||||
|
|
||||||
|
func _clear_tree():
|
||||||
|
signal_tree.clear()
|
|
@ -0,0 +1,78 @@
|
||||||
|
[gd_scene load_steps=5 format=3 uid="uid://dppfamjc0ji40"]
|
||||||
|
|
||||||
|
[ext_resource type="Script" path="res://addons/SignalVisualizer/Visualizer/signal_visualizer_dock.gd" id="1_akar5"]
|
||||||
|
[ext_resource type="Texture2D" uid="uid://bmnff63evbdhv" path="res://addons/SignalVisualizer/Clear.svg" id="2_m8bsv"]
|
||||||
|
[ext_resource type="Texture2D" uid="uid://bxj8ep08wbnm6" path="res://addons/SignalVisualizer/GraphEdit.svg" id="3_dtmqs"]
|
||||||
|
|
||||||
|
[sub_resource type="StyleBoxEmpty" id="StyleBoxEmpty_ae0jg"]
|
||||||
|
|
||||||
|
[node name="SignalVisualizerDock" type="Control"]
|
||||||
|
clip_contents = true
|
||||||
|
custom_minimum_size = Vector2(2.08165e-12, 200)
|
||||||
|
layout_mode = 3
|
||||||
|
anchors_preset = 15
|
||||||
|
anchor_right = 1.0
|
||||||
|
anchor_bottom = 1.0
|
||||||
|
grow_horizontal = 2
|
||||||
|
grow_vertical = 2
|
||||||
|
script = ExtResource("1_akar5")
|
||||||
|
|
||||||
|
[node name="VBoxContainer" type="VBoxContainer" parent="."]
|
||||||
|
clip_contents = true
|
||||||
|
layout_mode = 1
|
||||||
|
anchors_preset = 15
|
||||||
|
anchor_right = 1.0
|
||||||
|
anchor_bottom = 1.0
|
||||||
|
grow_horizontal = 2
|
||||||
|
grow_vertical = 2
|
||||||
|
size_flags_horizontal = 3
|
||||||
|
|
||||||
|
[node name="HBoxContainer" type="HBoxContainer" parent="VBoxContainer"]
|
||||||
|
clip_contents = true
|
||||||
|
custom_minimum_size = Vector2(2.08165e-12, 50)
|
||||||
|
layout_mode = 2
|
||||||
|
theme_override_constants/separation = 8
|
||||||
|
alignment = 2
|
||||||
|
|
||||||
|
[node name="ArrangeNodesCheckBox" type="CheckBox" parent="VBoxContainer/HBoxContainer"]
|
||||||
|
unique_name_in_owner = true
|
||||||
|
layout_mode = 2
|
||||||
|
text = "Arrange Nodes"
|
||||||
|
|
||||||
|
[node name="SignalDetailsCheckBox" type="CheckBox" parent="VBoxContainer/HBoxContainer"]
|
||||||
|
unique_name_in_owner = true
|
||||||
|
layout_mode = 2
|
||||||
|
text = "Signal Details"
|
||||||
|
|
||||||
|
[node name="Panel" type="Panel" parent="VBoxContainer/HBoxContainer"]
|
||||||
|
layout_mode = 2
|
||||||
|
size_flags_horizontal = 3
|
||||||
|
theme_override_styles/panel = SubResource("StyleBoxEmpty_ae0jg")
|
||||||
|
|
||||||
|
[node name="ClearGraphButton" type="Button" parent="VBoxContainer/HBoxContainer"]
|
||||||
|
layout_mode = 2
|
||||||
|
text = "Clear Graph"
|
||||||
|
icon = ExtResource("2_m8bsv")
|
||||||
|
|
||||||
|
[node name="GenerateGraphButton" type="Button" parent="VBoxContainer/HBoxContainer"]
|
||||||
|
layout_mode = 2
|
||||||
|
text = "Generate Graph"
|
||||||
|
icon = ExtResource("3_dtmqs")
|
||||||
|
|
||||||
|
[node name="HSplitContainer" type="HSplitContainer" parent="VBoxContainer"]
|
||||||
|
layout_mode = 2
|
||||||
|
size_flags_vertical = 3
|
||||||
|
|
||||||
|
[node name="SignalTree" type="Tree" parent="VBoxContainer/HSplitContainer"]
|
||||||
|
unique_name_in_owner = true
|
||||||
|
custom_minimum_size = Vector2(200, 2.08165e-12)
|
||||||
|
layout_mode = 2
|
||||||
|
column_titles_visible = true
|
||||||
|
|
||||||
|
[node name="Graph" type="GraphEdit" parent="VBoxContainer/HSplitContainer"]
|
||||||
|
unique_name_in_owner = true
|
||||||
|
layout_mode = 2
|
||||||
|
size_flags_vertical = 3
|
||||||
|
|
||||||
|
[connection signal="pressed" from="VBoxContainer/HBoxContainer/ClearGraphButton" to="." method="_on_clear_graph_button_pressed"]
|
||||||
|
[connection signal="pressed" from="VBoxContainer/HBoxContainer/GenerateGraphButton" to="." method="_on_generate_graph_button_pressed"]
|
7
addons/SignalVisualizer/plugin.cfg
Normal file
|
@ -0,0 +1,7 @@
|
||||||
|
[plugin]
|
||||||
|
|
||||||
|
name="SignalVisualizer"
|
||||||
|
description="Visual the current scene's signal connections as a graph. Debug the current running scene's signals with automatic logging in a new debugger panel."
|
||||||
|
author="MiniGameDev"
|
||||||
|
version="1.7.0"
|
||||||
|
script="SignalVisualizer.gd"
|
|
@ -17,6 +17,7 @@ func _on_duck_gameover() -> void:
|
||||||
show()
|
show()
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
func _on_restart_pressed() -> void:
|
func _on_restart_pressed() -> void:
|
||||||
$"../..".start_game()
|
$"../..".start_game()
|
||||||
$AnimationPlayer.play("out-in")
|
$AnimationPlayer.play("out-in")
|
||||||
|
|
3
code/first_save.json
Normal file
|
@ -0,0 +1,3 @@
|
||||||
|
{
|
||||||
|
"hiscore":0
|
||||||
|
}
|
39
code/global.gd
Normal file
|
@ -0,0 +1,39 @@
|
||||||
|
extends Node
|
||||||
|
|
||||||
|
|
||||||
|
signal levelup
|
||||||
|
signal start
|
||||||
|
|
||||||
|
var maxspeed = 260
|
||||||
|
var speed = 100
|
||||||
|
var score = 0
|
||||||
|
var gamerunning = false
|
||||||
|
|
||||||
|
var hiscore
|
||||||
|
|
||||||
|
func _ready() -> void:
|
||||||
|
load_save()
|
||||||
|
#write_save()
|
||||||
|
|
||||||
|
func load_save():
|
||||||
|
var data : FileAccess
|
||||||
|
if (not FileAccess.file_exists("user://save.json")) or FileAccess.open("user://save.json",FileAccess.READ_WRITE).get_as_text() == "":
|
||||||
|
data = FileAccess.open("user://save.json",FileAccess.WRITE)
|
||||||
|
data.store_string(FileAccess.open("res://code/first_save.json",FileAccess.READ).get_as_text())
|
||||||
|
data.close()
|
||||||
|
data = FileAccess.open("user://save.json",FileAccess.READ)
|
||||||
|
var contenttext = JSON.parse_string(data.get_as_text())
|
||||||
|
print(contenttext)
|
||||||
|
hiscore = contenttext["hiscore"]
|
||||||
|
|
||||||
|
|
||||||
|
func write_save():
|
||||||
|
var data : FileAccess
|
||||||
|
data = FileAccess.open("user://save.json",FileAccess.WRITE)
|
||||||
|
data.store_string(
|
||||||
|
JSON.stringify(
|
||||||
|
{
|
||||||
|
"hiscore":$"/root/Game/Gui/HiscoreContainer/hiscore".get_score()
|
||||||
|
}
|
||||||
|
)
|
||||||
|
)
|
18
code/hiscore.gd
Normal file
|
@ -0,0 +1,18 @@
|
||||||
|
extends Label
|
||||||
|
|
||||||
|
|
||||||
|
# Called when the node enters the scene tree for the first time.
|
||||||
|
func _ready() -> void:
|
||||||
|
text = str($"/root/Global".hiscore)
|
||||||
|
|
||||||
|
# Called every frame. 'delta' is the elapsed time since the previous frame.
|
||||||
|
func _process(delta: float) -> void:
|
||||||
|
pass
|
||||||
|
|
||||||
|
func set_new_hiscore():
|
||||||
|
if int(text) < int(%ScoreContainer/text.text):
|
||||||
|
text = %ScoreContainer/text.text
|
||||||
|
$"/root/Global".write_save()
|
||||||
|
|
||||||
|
func get_score():
|
||||||
|
return int(text)
|
11
code/main.gd
|
@ -5,6 +5,8 @@ var biom_n_w = preload("res://assets/Ground/normal_wueste.png")
|
||||||
var biom_w = preload("res://assets/Ground/wueste.png")
|
var biom_w = preload("res://assets/Ground/wueste.png")
|
||||||
func _ready() -> void:
|
func _ready() -> void:
|
||||||
$Screen/Background/Floor/EasterEggFixCollision.disabled = false
|
$Screen/Background/Floor/EasterEggFixCollision.disabled = false
|
||||||
|
$Gui/HiscoreContainer.show()
|
||||||
|
%ScoreContainer.hide()
|
||||||
|
|
||||||
func start_game():
|
func start_game():
|
||||||
$Screen/Background/Floor/EasterEggFixCollision.disabled = true
|
$Screen/Background/Floor/EasterEggFixCollision.disabled = true
|
||||||
|
@ -13,6 +15,8 @@ func start_game():
|
||||||
$Screen/Duck.start()
|
$Screen/Duck.start()
|
||||||
$Screen/HindernissSpawner.del_all()
|
$Screen/HindernissSpawner.del_all()
|
||||||
$Screen/EastereggSpawner.del_all()
|
$Screen/EastereggSpawner.del_all()
|
||||||
|
$Gui/HiscoreContainer.hide()
|
||||||
|
%ScoreContainer.show()
|
||||||
$"/root/Global".gamerunning = true
|
$"/root/Global".gamerunning = true
|
||||||
$"/root/Global".start.emit()
|
$"/root/Global".start.emit()
|
||||||
$"/root/Global".score = 0
|
$"/root/Global".score = 0
|
||||||
|
@ -25,17 +29,14 @@ func _on_level_up_timer_timeout() -> void:
|
||||||
|
|
||||||
func _score() -> void:
|
func _score() -> void:
|
||||||
if not $"/root/Global".gamerunning:
|
if not $"/root/Global".gamerunning:
|
||||||
$Gui/Score.text = ""
|
%ScoreContainer/score.text = ""
|
||||||
elif $Screen/Duck.alive:
|
elif $Screen/Duck.alive:
|
||||||
$"/root/Global".score += 10
|
$"/root/Global".score += 10
|
||||||
$Gui/Score.text = "Score: "+str($"/root/Global".score)
|
%ScoreContainer/score.text = str($"/root/Global".score)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
func _process(delta):
|
func _process(delta):
|
||||||
var background_size_x = $Screen/Background/Sprite.texture.get_size().x
|
var background_size_x = $Screen/Background/Sprite.texture.get_size().x
|
||||||
|
|
||||||
|
|
||||||
var bla = (background_size_x) + $Screen/Background/Sprite.position.x
|
var bla = (background_size_x) + $Screen/Background/Sprite.position.x
|
||||||
if $"/root/Global".gamerunning:
|
if $"/root/Global".gamerunning:
|
||||||
$Screen/Background/Sprite.position.x -= $"/root/Global".speed*delta
|
$Screen/Background/Sprite.position.x -= $"/root/Global".speed*delta
|
||||||
|
|
|
@ -1,9 +0,0 @@
|
||||||
extends Label
|
|
||||||
|
|
||||||
var speed = 70
|
|
||||||
# Called when the node enters the scene tree for the first time.
|
|
||||||
#func _ready() -> void:
|
|
||||||
#text = "0"
|
|
||||||
|
|
||||||
|
|
||||||
# Called every frame. 'delta' is the elapsed time since the previous frame.
|
|
|
@ -9,7 +9,7 @@ custom_features=""
|
||||||
export_filter="all_resources"
|
export_filter="all_resources"
|
||||||
include_filter=""
|
include_filter=""
|
||||||
exclude_filter=""
|
exclude_filter=""
|
||||||
export_path="builds/DuckRun_v2.1.0.apk"
|
export_path="builds/DuckRun_v2.1.0beta.apk"
|
||||||
encryption_include_filters=""
|
encryption_include_filters=""
|
||||||
encryption_exclude_filters=""
|
encryption_exclude_filters=""
|
||||||
encrypt_pck=false
|
encrypt_pck=false
|
||||||
|
@ -42,8 +42,8 @@ package/exclude_from_recents=false
|
||||||
package/show_in_android_tv=true
|
package/show_in_android_tv=true
|
||||||
package/show_in_app_library=true
|
package/show_in_app_library=true
|
||||||
package/show_as_launcher_app=false
|
package/show_as_launcher_app=false
|
||||||
launcher_icons/main_192x192="res://icon.png"
|
launcher_icons/main_192x192=""
|
||||||
launcher_icons/adaptive_foreground_432x432=""
|
launcher_icons/adaptive_foreground_432x432="res://icon.png"
|
||||||
launcher_icons/adaptive_background_432x432=""
|
launcher_icons/adaptive_background_432x432=""
|
||||||
graphics/opengl_debug=false
|
graphics/opengl_debug=false
|
||||||
xr_features/xr_mode=0
|
xr_features/xr_mode=0
|
||||||
|
|
1
icon.svg
Normal file
|
@ -0,0 +1 @@
|
||||||
|
<svg height="128" width="128" xmlns="http://www.w3.org/2000/svg"><rect x="2" y="2" width="124" height="124" rx="14" fill="#363d52" stroke="#212532" stroke-width="4"/><g transform="scale(.101) translate(122 122)"><g fill="#fff"><path d="M105 673v33q407 354 814 0v-33z"/><path fill="#478cbf" d="m105 673 152 14q12 1 15 14l4 67 132 10 8-61q2-11 15-15h162q13 4 15 15l8 61 132-10 4-67q3-13 15-14l152-14V427q30-39 56-81-35-59-83-108-43 20-82 47-40-37-88-64 7-51 8-102-59-28-123-42-26 43-46 89-49-7-98 0-20-46-46-89-64 14-123 42 1 51 8 102-48 27-88 64-39-27-82-47-48 49-83 108 26 42 56 81zm0 33v39c0 276 813 276 813 0v-39l-134 12-5 69q-2 10-14 13l-162 11q-12 0-16-11l-10-65H447l-10 65q-4 11-16 11l-162-11q-12-3-14-13l-5-69z"/><path d="M483 600c3 34 55 34 58 0v-86c-3-34-55-34-58 0z"/><circle cx="725" cy="526" r="90"/><circle cx="299" cy="526" r="90"/></g><g fill="#414042"><circle cx="307" cy="532" r="60"/><circle cx="717" cy="532" r="60"/></g></g></svg>
|
After Width: | Height: | Size: 950 B |
37
icon.svg.import
Normal file
|
@ -0,0 +1,37 @@
|
||||||
|
[remap]
|
||||||
|
|
||||||
|
importer="texture"
|
||||||
|
type="CompressedTexture2D"
|
||||||
|
uid="uid://bw8oucm2thxee"
|
||||||
|
path="res://.godot/imported/icon.svg-218a8f2b3041327d8a5756f3a245f83b.ctex"
|
||||||
|
metadata={
|
||||||
|
"vram_texture": false
|
||||||
|
}
|
||||||
|
|
||||||
|
[deps]
|
||||||
|
|
||||||
|
source_file="res://icon.svg"
|
||||||
|
dest_files=["res://.godot/imported/icon.svg-218a8f2b3041327d8a5756f3a245f83b.ctex"]
|
||||||
|
|
||||||
|
[params]
|
||||||
|
|
||||||
|
compress/mode=0
|
||||||
|
compress/high_quality=false
|
||||||
|
compress/lossy_quality=0.7
|
||||||
|
compress/hdr_compression=1
|
||||||
|
compress/normal_map=0
|
||||||
|
compress/channel_pack=0
|
||||||
|
mipmaps/generate=false
|
||||||
|
mipmaps/limit=-1
|
||||||
|
roughness/mode=0
|
||||||
|
roughness/src_normal=""
|
||||||
|
process/fix_alpha_border=true
|
||||||
|
process/premult_alpha=false
|
||||||
|
process/normal_map_invert_y=false
|
||||||
|
process/hdr_as_srgb=false
|
||||||
|
process/hdr_clamp_exposure=false
|
||||||
|
process/size_limit=0
|
||||||
|
detect_3d/compress_to=1
|
||||||
|
svg/scale=1.0
|
||||||
|
editor/scale_with_editor_scale=false
|
||||||
|
editor/convert_colors_with_editor_theme=false
|
0
images/.gdignore
Normal file
BIN
images/AutoloadScreenshot.png
Normal file
After Width: | Height: | Size: 314 KiB |
BIN
images/IncomingSignalScreenshot.png
Normal file
After Width: | Height: | Size: 100 KiB |
BIN
images/OutgoingSignalScreenshot .png
Normal file
After Width: | Height: | Size: 28 KiB |
BIN
images/PluginScreenshot.png
Normal file
After Width: | Height: | Size: 310 KiB |
BIN
images/SignalDebuggerDataScreenshot.png
Normal file
After Width: | Height: | Size: 249 KiB |
BIN
images/SignalDebuggerPanel.png
Normal file
After Width: | Height: | Size: 87 KiB |
BIN
images/SignalDebuggerSignalLogScreenshot.png
Normal file
After Width: | Height: | Size: 115 KiB |
BIN
images/SignalDebuggerSignalTreeScreenshot.png
Normal file
After Width: | Height: | Size: 64 KiB |
BIN
images/SignalVisualizerDemo.png
Normal file
After Width: | Height: | Size: 1.5 MiB |
BIN
images/SignalVisualizerDockScreenshot.png
Normal file
After Width: | Height: | Size: 18 KiB |
BIN
images/SignalVisualizerGraphDemo.png
Normal file
After Width: | Height: | Size: 374 KiB |
BIN
images/SignalVisualizerIcon.png
Normal file
After Width: | Height: | Size: 2.9 KiB |
BIN
images/SignalVisualizerToolbarScreenshot.png
Normal file
After Width: | Height: | Size: 11 KiB |
BIN
m5x7.ttf
|
@ -19,7 +19,7 @@ config/icon="res://icon.png"
|
||||||
|
|
||||||
[autoload]
|
[autoload]
|
||||||
|
|
||||||
Global="*res://scenes/global.gd"
|
Global="*res://code/global.gd"
|
||||||
|
|
||||||
[display]
|
[display]
|
||||||
|
|
||||||
|
@ -36,13 +36,17 @@ window/handheld/orientation=4
|
||||||
version_control/plugin_name="GitPlugin"
|
version_control/plugin_name="GitPlugin"
|
||||||
version_control/autoload_on_startup=true
|
version_control/autoload_on_startup=true
|
||||||
|
|
||||||
|
[editor_plugins]
|
||||||
|
|
||||||
|
enabled=PackedStringArray("res://addons/SignalVisualizer/plugin.cfg")
|
||||||
|
|
||||||
[file_customization]
|
[file_customization]
|
||||||
|
|
||||||
folder_colors={
|
folder_colors={
|
||||||
"res://android/": "gray",
|
"res://android/": "gray",
|
||||||
"res://assets/": "green",
|
"res://assets/": "green",
|
||||||
"res://builds/": "gray",
|
"res://builds/": "gray",
|
||||||
"res://code/": "blue",
|
"res://code/": "purple",
|
||||||
"res://scenes/": "orange"
|
"res://scenes/": "orange"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
204
scenes/Main.tscn
|
@ -1,10 +0,0 @@
|
||||||
extends Node
|
|
||||||
|
|
||||||
|
|
||||||
signal levelup
|
|
||||||
signal start
|
|
||||||
|
|
||||||
var maxspeed = 260
|
|
||||||
var speed = 100
|
|
||||||
var score = 0
|
|
||||||
var gamerunning = false
|
|