You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
225 lines
8.1 KiB
225 lines
8.1 KiB
3 years ago
|
tool
|
||
|
extends Control
|
||
|
|
||
|
# Don't change this if possible
|
||
|
export (String) var documentation_path : String = "res://addons/dialogic/Documentation"
|
||
|
|
||
|
# This enables/disables the use of folder files
|
||
|
# If enabled, the docs will expect a file named
|
||
|
# exactly like a folder for each folder in the docs:
|
||
|
## E.g.: If you have a Tutorials folder somewhere put a Tutorials.md file next to it.
|
||
|
## This way the folder will be clickable and you can see the page,
|
||
|
## but it won't be shown as a separate page
|
||
|
var use_folder_files = true
|
||
|
|
||
|
# These files will not be listed. Just use the filename! No paths in here
|
||
|
var file_ignore_list = ['Welcome.md']
|
||
|
|
||
|
|
||
|
################################################################################
|
||
|
## PUBLIC FUNCTIONS ##
|
||
|
################################################################################
|
||
|
|
||
|
## Returns a dictionary that contains the important parts of the
|
||
|
## documentations Content folder.
|
||
|
##
|
||
|
## This is mainly used if you want to somehow display a list of the docs content,
|
||
|
## for example to create a file-tree or a list of documents
|
||
|
##
|
||
|
## Only files ending on .md are noticed.
|
||
|
## Folders that contain no such files are ignored
|
||
|
func get_documentation_content():
|
||
|
return get_dir_contents(documentation_path+"/Content")
|
||
|
|
||
|
## Will create a hirarchy of TreeItems on the given 'trees' root_item
|
||
|
## If not root_item is given a new root_item will be created
|
||
|
## The root item does not have to be the actual root item of the whole tree,
|
||
|
## but the root of the documentation branch.
|
||
|
##
|
||
|
## With def_folder_info and def_page_info special information can be
|
||
|
## added to the meta of the Items
|
||
|
##
|
||
|
## If a filter_term is given, only items with that filter will be created.
|
||
|
## Right now there will always be all folders.
|
||
|
func build_documentation_tree(tree : Tree, root_item:TreeItem = null, def_folder_info:Dictionary = {}, def_page_info:Dictionary = {}, filter_term:String = ''):
|
||
|
return _build_documentation_tree(tree, root_item, def_folder_info, def_page_info, filter_term)
|
||
|
|
||
|
|
||
|
################################################################################
|
||
|
## PRIVATE FUNCTIONS ##
|
||
|
################################################################################
|
||
|
|
||
|
|
||
|
#~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||
|
### LOOKING THROUGH THE DOCS FOLDERS:
|
||
|
|
||
|
func get_dir_contents(rootPath: String) -> Dictionary:
|
||
|
var directory_structure = {}
|
||
|
var dir := Directory.new()
|
||
|
|
||
|
if dir.open(rootPath) == OK:
|
||
|
dir.list_dir_begin(true, false)
|
||
|
directory_structure = _add_dir_contents(dir)
|
||
|
else:
|
||
|
push_error("Docs: An error occurred when trying to access the path.")
|
||
|
return directory_structure
|
||
|
|
||
|
func _add_dir_contents(dir: Directory) -> Dictionary:
|
||
|
var file_name = dir.get_next()
|
||
|
|
||
|
var structure = {}
|
||
|
while (file_name != ""):
|
||
|
var path = dir.get_current_dir() + "/" + file_name
|
||
|
if dir.current_is_dir():
|
||
|
#print("Found directory: %s" % path)
|
||
|
var subDir = Directory.new()
|
||
|
subDir.open(path)
|
||
|
subDir.list_dir_begin(true, false)
|
||
|
var dir_content = _add_dir_contents(subDir)
|
||
|
if dir_content.has('_files_'):
|
||
|
structure[path] = dir_content
|
||
|
else:
|
||
|
#print("Found file: %s" % path)
|
||
|
if not file_name.ends_with(".md"):
|
||
|
file_name = dir.get_next()
|
||
|
continue
|
||
|
if file_name in file_ignore_list:
|
||
|
file_name = dir.get_next()
|
||
|
continue
|
||
|
if not structure.has("_files_"):
|
||
|
structure["_files_"] = []
|
||
|
|
||
|
structure["_files_"].append(path)
|
||
|
|
||
|
file_name = dir.get_next()
|
||
|
dir.list_dir_end()
|
||
|
return structure
|
||
|
|
||
|
#~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||
|
### For bouilding the tree
|
||
|
|
||
|
func _build_documentation_tree(tree : Tree, root_item:TreeItem = null, def_folder_info:Dictionary = {}, def_page_info:Dictionary = {}, filter_term:String =''):
|
||
|
|
||
|
var documentation_tree
|
||
|
if root_item == null:
|
||
|
documentation_tree = tree.create_item()
|
||
|
documentation_tree.set_text(0, "Documentation")
|
||
|
|
||
|
else:
|
||
|
documentation_tree = root_item
|
||
|
|
||
|
# if no search is performed, collapse the tree by default
|
||
|
if not filter_term:
|
||
|
documentation_tree.collapsed = true
|
||
|
else:
|
||
|
documentation_tree.collapsed = false
|
||
|
|
||
|
# create the rest of the tree based on the dict we get from the DocsHelper
|
||
|
var doc_structure = get_documentation_content()
|
||
|
#print(doc_structure)
|
||
|
create_doc_tree(tree, documentation_tree, def_folder_info, def_page_info, doc_structure, filter_term)
|
||
|
return documentation_tree
|
||
|
|
||
|
# this calls itself recursivly to create the tree, based on the given dict
|
||
|
func create_doc_tree(tree, parent_item, def_folder_info, def_page_info, doc_structure, filter_term):
|
||
|
for key in doc_structure.keys():
|
||
|
# if this is a folder
|
||
|
if typeof(doc_structure[key]) == TYPE_DICTIONARY:
|
||
|
var folder_item = _add_documentation_folder(tree, parent_item, {'name':key.get_file(), 'path':key}, def_folder_info)
|
||
|
create_doc_tree(tree, folder_item, def_folder_info, def_page_info, doc_structure[key], filter_term)
|
||
|
if not filter_term:
|
||
|
folder_item.collapsed = true
|
||
|
# if this is a page
|
||
|
elif typeof(doc_structure[key]) == TYPE_ARRAY:
|
||
|
for file in doc_structure[key]:
|
||
|
if use_folder_files and file.trim_suffix('.md') in doc_structure.keys():
|
||
|
pass
|
||
|
else:
|
||
|
if not filter_term or (filter_term and filter_term.to_lower() in get_title(file, '').to_lower()):
|
||
|
_add_documentation_page(tree, parent_item, {'name':file.get_file().trim_suffix(".md"), 'path': file}, def_page_info)
|
||
|
|
||
|
func merge_dir(target: Dictionary, patch: Dictionary):
|
||
|
var copy = target.duplicate()
|
||
|
for key in patch:
|
||
|
copy[key] = patch[key]
|
||
|
return copy
|
||
|
|
||
|
# this adds a folder item to the tree
|
||
|
func _add_documentation_folder(tree, parent_item, folder_info, default_info):
|
||
|
var item = tree.create_item(parent_item)
|
||
|
item.set_text(0, folder_info['name'])
|
||
|
item.set_icon(0, tree.get_icon("HelpSearch", "EditorIcons"))
|
||
|
item.set_editable(0, false)
|
||
|
if use_folder_files:
|
||
|
var x = File.new()
|
||
|
if x.file_exists(folder_info['path']+'.md'):
|
||
|
folder_info['path'] += '.md'
|
||
|
else:
|
||
|
folder_info['path'] = ''
|
||
|
else:
|
||
|
folder_info['path'] = ''
|
||
|
item.set_metadata(0, merge_dir(default_info, folder_info))
|
||
|
if not tree.get_constant("dark_theme", "Editor"):
|
||
|
item.set_icon_modulate(0, get_color("property_color", "Editor"))
|
||
|
return item
|
||
|
|
||
|
# this adds a page item to the tree
|
||
|
func _add_documentation_page(tree, parent, page_info, default_info):
|
||
|
var item = tree.create_item(parent)
|
||
|
item.set_text(0, get_title(page_info['path'], page_info['name']))
|
||
|
item.set_tooltip(0,page_info['path'])
|
||
|
item.set_editable(0, false)
|
||
|
item.set_icon(0, tree.get_icon("Help", "EditorIcons"))
|
||
|
var new_dir = merge_dir(default_info, page_info)
|
||
|
#print(new_dir)
|
||
|
item.set_metadata(0,new_dir)
|
||
|
if not tree.get_constant("dark_theme", "Editor"):
|
||
|
item.set_icon_modulate(0, get_color("property_color", "Editor"))
|
||
|
return item
|
||
|
|
||
|
# returns the first line of a text_file, a bit cleaned up
|
||
|
func get_title(path, default_name):
|
||
|
# opening the file
|
||
|
var f = File.new()
|
||
|
f.open(path, File.READ)
|
||
|
var arr = f.get_as_text().split('\n', false, 1)
|
||
|
if not arr.empty():
|
||
|
return arr[0].replace('#', '').strip_edges()
|
||
|
else:
|
||
|
return default_name
|
||
|
## ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||
|
## For searching the tree
|
||
|
## used to search and select an item of the tree based on a info saved in the metadata
|
||
|
## in most cases you just want to search for the item that has a certain path
|
||
|
##
|
||
|
## the paren_item parameter is only used so this can call itself recursivly
|
||
|
func search_and_select_docs(docs_tree_item:TreeItem, info:String, key:String = 'path'):
|
||
|
if info == "": return
|
||
|
if info == "/":
|
||
|
docs_tree_item.select(0)
|
||
|
return true
|
||
|
#print("Asearch ", key, " ", info)
|
||
|
#print("Asearchin on item: ", docs_tree_item.get_text(0))
|
||
|
var item = docs_tree_item.get_children()
|
||
|
while item:
|
||
|
#print("A ",item.get_text(0))
|
||
|
if not item.has_method('get_metadata'):
|
||
|
item = item.get_next()
|
||
|
|
||
|
var meta = item.get_metadata(0)
|
||
|
#print(meta)
|
||
|
if meta.has(key):
|
||
|
if meta[key] == info:
|
||
|
item.select(0)
|
||
|
return true
|
||
|
if search_and_select_docs(item, info, key):
|
||
|
return true
|
||
|
item = item.get_next()
|
||
|
return false
|
||
|
|
||
|
#~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||
|
#### For bouilding the tree
|
||
|
#func create_reference():
|
||
|
# var RefColl = ReferenceCollector.new()
|
||
|
# RefColl._run()
|