2019-12-20 20:22:52 +00:00
|
|
|
extends Node
|
|
|
|
|
2020-04-11 22:36:58 +00:00
|
|
|
# Get hold of the brushes, including random brushes (subdirectories and % files
|
|
|
|
# in them, non % files get loaded independently.) nyaaa
|
|
|
|
# Returns a list of [
|
2020-04-12 22:40:26 +00:00
|
|
|
# [non random single png files in the root subdir],
|
2020-04-11 22:36:58 +00:00
|
|
|
# {
|
2020-04-12 22:40:26 +00:00
|
|
|
# map of subdirectories to lists of files for
|
2020-04-11 22:36:58 +00:00
|
|
|
# the randomised brush - if a directory contains no
|
|
|
|
# randomised files then it is not included in this.
|
|
|
|
# },
|
|
|
|
# {
|
|
|
|
# map of subdirectories to lists of files inside of them
|
|
|
|
# that are not for randomised brushes.
|
|
|
|
# }
|
|
|
|
# ]
|
2020-04-12 22:40:26 +00:00
|
|
|
# The separation of nonrandomised and randomised files
|
2020-04-11 22:36:58 +00:00
|
|
|
# in subdirectories allows different XDG_DATA_DIR overriding
|
|
|
|
# for each nyaa.
|
|
|
|
#
|
|
|
|
# Returns null if the directory gave an error opening.
|
2020-04-12 22:40:26 +00:00
|
|
|
#
|
2020-04-11 22:36:58 +00:00
|
|
|
func get_brush_files_from_directory(directory: String): # -> Array
|
|
|
|
var base_png_files := [] # list of files in the base directory
|
|
|
|
var subdirectories := [] # list of subdirectories to process.
|
2020-04-12 22:40:26 +00:00
|
|
|
|
|
|
|
var randomised_subdir_files_map : Dictionary = {}
|
2020-04-11 22:36:58 +00:00
|
|
|
var nonrandomised_subdir_files_map : Dictionary = {}
|
2020-04-12 22:40:26 +00:00
|
|
|
|
2020-04-11 22:36:58 +00:00
|
|
|
var main_directory : Directory = Directory.new()
|
|
|
|
var err := main_directory.open(directory)
|
|
|
|
if err != OK:
|
|
|
|
return null
|
2020-04-12 22:40:26 +00:00
|
|
|
|
|
|
|
|
2020-04-11 22:36:58 +00:00
|
|
|
# Build first the list of base png files and all subdirectories to
|
|
|
|
# scan later (skip navigational . and ..)
|
|
|
|
main_directory.list_dir_begin(true)
|
|
|
|
var fname : String = main_directory.get_next()
|
|
|
|
while fname != "":
|
|
|
|
if main_directory.current_is_dir():
|
|
|
|
subdirectories.append(fname)
|
|
|
|
else: # Filter for pngs
|
|
|
|
if fname.get_extension().to_lower() == "png":
|
|
|
|
print_debug("Found brush at '%s'" % [fname])
|
|
|
|
base_png_files.append(fname)
|
2020-04-12 22:40:26 +00:00
|
|
|
|
2020-04-11 22:36:58 +00:00
|
|
|
# go to next
|
|
|
|
fname = main_directory.get_next()
|
|
|
|
main_directory.list_dir_end()
|
2020-04-12 22:40:26 +00:00
|
|
|
|
2020-04-11 22:36:58 +00:00
|
|
|
# Now we iterate over subdirectories!
|
|
|
|
for subdirectory in subdirectories:
|
|
|
|
var the_directory : Directory = Directory.new()
|
2020-04-12 22:40:26 +00:00
|
|
|
|
2020-04-11 22:36:58 +00:00
|
|
|
# Holds names of files that make this
|
|
|
|
# a component of a randomised brush ^.^
|
|
|
|
var randomised_files := []
|
2020-04-12 22:40:26 +00:00
|
|
|
|
2020-04-11 22:36:58 +00:00
|
|
|
# Non-randomise-indicated image files
|
|
|
|
var non_randomised_files := []
|
2020-04-12 22:40:26 +00:00
|
|
|
|
2020-04-11 22:36:58 +00:00
|
|
|
the_directory.open(directory.plus_file(subdirectory))
|
|
|
|
the_directory.list_dir_begin(true)
|
|
|
|
var curr_file := the_directory.get_next()
|
2020-04-12 22:40:26 +00:00
|
|
|
|
2020-04-11 22:36:58 +00:00
|
|
|
while curr_file != "":
|
|
|
|
# only do stuff if we are actually dealing with a file
|
|
|
|
# and png one at that nya
|
|
|
|
if !the_directory.current_is_dir() and curr_file.get_extension().to_lower() == "png":
|
|
|
|
# if we are a random element, add
|
|
|
|
if "%" in curr_file:
|
|
|
|
randomised_files.append(curr_file)
|
|
|
|
else:
|
|
|
|
non_randomised_files.append(curr_file)
|
|
|
|
curr_file = the_directory.get_next()
|
2020-04-12 22:40:26 +00:00
|
|
|
|
2020-04-11 22:36:58 +00:00
|
|
|
the_directory.list_dir_end()
|
2020-04-12 22:40:26 +00:00
|
|
|
|
2020-04-11 22:36:58 +00:00
|
|
|
# Add these to the maps nyaa
|
|
|
|
if len(randomised_files) > 0:
|
|
|
|
randomised_subdir_files_map[subdirectory] = randomised_files
|
|
|
|
if len(non_randomised_files) > 0:
|
|
|
|
nonrandomised_subdir_files_map[subdirectory] = non_randomised_files
|
|
|
|
# We are done generating the maps!
|
|
|
|
return [base_png_files, randomised_subdir_files_map, nonrandomised_subdir_files_map]
|
|
|
|
|
|
|
|
|
|
|
|
# Add a randomised brush from the given list of files as a
|
2020-04-12 22:40:26 +00:00
|
|
|
# source.
|
|
|
|
# The tooltip name is what shows up on the tooltip
|
2020-04-11 22:36:58 +00:00
|
|
|
# and is probably in this case the name of the containing
|
|
|
|
# randomised directory.
|
|
|
|
func add_randomised_brush(fpaths : Array, tooltip_name : String) -> void:
|
|
|
|
# Attempt to load the images from the file paths.
|
2020-04-12 22:40:26 +00:00
|
|
|
var loaded_images : Array = []
|
2020-04-11 22:36:58 +00:00
|
|
|
for filen in fpaths:
|
|
|
|
var image := Image.new()
|
|
|
|
var err := image.load(filen)
|
|
|
|
if err == OK:
|
|
|
|
image.convert(Image.FORMAT_RGBA8)
|
|
|
|
loaded_images.append(image)
|
|
|
|
print_debug("Loaded image from '%s'" % [filen])
|
2020-04-12 22:40:26 +00:00
|
|
|
|
2020-04-11 22:36:58 +00:00
|
|
|
# If any images were successfully loaded, then
|
|
|
|
# we create the randomised brush button, copied
|
|
|
|
# from find_brushes.
|
2020-04-12 22:40:26 +00:00
|
|
|
|
2020-04-11 22:36:58 +00:00
|
|
|
if len(loaded_images) > 0: # actually have images
|
|
|
|
# to use.
|
|
|
|
# take initial image...
|
|
|
|
var first_image : Image = loaded_images.pop_front()
|
2020-04-12 22:40:26 +00:00
|
|
|
|
2020-04-11 22:36:58 +00:00
|
|
|
# The index which this random brush will be at
|
|
|
|
var next_random_brush_index := Global.file_brush_container.get_child_count()
|
2020-04-12 22:40:26 +00:00
|
|
|
|
2020-04-11 22:36:58 +00:00
|
|
|
Global.custom_brushes.append(first_image)
|
|
|
|
Global.create_brush_button(first_image, Global.Brush_Types.RANDOM_FILE, tooltip_name)
|
|
|
|
# # Process the rest
|
|
|
|
for remaining_image in loaded_images:
|
|
|
|
var brush_button = Global.file_brush_container.get_child(next_random_brush_index)
|
|
|
|
brush_button.random_brushes.append(remaining_image)
|
|
|
|
|
|
|
|
# Add a plain brush from the given path to the list of brushes.
|
|
|
|
# Taken, again, from find_brushes
|
|
|
|
func add_plain_brush(path: String, tooltip_name: String) -> void:
|
|
|
|
var image := Image.new()
|
|
|
|
var err := image.load(path)
|
|
|
|
if err != OK:
|
|
|
|
return
|
|
|
|
# do the standard conversion thing...
|
|
|
|
image.convert(Image.FORMAT_RGBA8)
|
|
|
|
Global.custom_brushes.append(image)
|
|
|
|
Global.create_brush_button(image, Global.Brush_Types.FILE, tooltip_name)
|
|
|
|
|
|
|
|
|
|
|
|
# Import brushes, in priority order, from the paths in question in priority order
|
|
|
|
# i.e. with an override system
|
|
|
|
# We use a very particular override system here where, for randomised brushes
|
|
|
|
# the directories containing them get overridden, but for nonrandomised files
|
2020-04-12 22:40:26 +00:00
|
|
|
# (including in directories containing randomised components too), the override
|
2020-04-11 22:36:58 +00:00
|
|
|
# is on a file-by-file basis nyaaaa ^.^
|
|
|
|
func import_brushes(priority_ordered_search_path: Array) -> void:
|
|
|
|
# Maps for files in the base directory (name : true)
|
|
|
|
var processed_basedir_paths : Dictionary = {}
|
|
|
|
var randomised_brush_subdirectories : Dictionary = {}
|
|
|
|
# Map from a subdirectory to a map similar to processed_basedir_files
|
|
|
|
# i.e. once a filename has been dealt with, set it to true.
|
|
|
|
var processed_subdir_paths : Dictionary = {}
|
2020-04-12 22:40:26 +00:00
|
|
|
|
2020-04-11 22:36:58 +00:00
|
|
|
# Sets of results of get_brush_files_from_directory
|
|
|
|
var all_available_paths : Array = []
|
|
|
|
for directory in priority_ordered_search_path:
|
|
|
|
all_available_paths.append(get_brush_files_from_directory(directory))
|
2020-04-12 22:40:26 +00:00
|
|
|
|
2020-04-11 22:36:58 +00:00
|
|
|
# Now to process. Note these are in order of the
|
|
|
|
# priority, as intended nyaa :)
|
|
|
|
for i in range(len(all_available_paths)):
|
|
|
|
var available_brush_file_information = all_available_paths[i]
|
|
|
|
var current_main_directory: String = priority_ordered_search_path[i]
|
|
|
|
if available_brush_file_information != null:
|
|
|
|
# The brush files in the main directory
|
|
|
|
var main_directory_file_paths : Array = available_brush_file_information[0]
|
2020-04-12 22:40:26 +00:00
|
|
|
# The subdirectory/list-of-randomised-brush-files
|
2020-04-11 22:36:58 +00:00
|
|
|
# map for this directory
|
|
|
|
var randomised_brush_subdirectory_map : Dictionary = available_brush_file_information[1]
|
|
|
|
# Map for subdirectories to non-randomised-brush files nyaa
|
|
|
|
var nonrandomised_brush_subdirectory_map : Dictionary = available_brush_file_information[2]
|
2020-04-12 22:40:26 +00:00
|
|
|
|
2020-04-11 22:36:58 +00:00
|
|
|
# Iterate over components and do stuff with them! nyaa
|
|
|
|
# first for the main directory path...
|
|
|
|
for subfile in main_directory_file_paths:
|
|
|
|
if not (subfile in processed_basedir_paths):
|
|
|
|
add_plain_brush(
|
2020-04-12 22:40:26 +00:00
|
|
|
current_main_directory.plus_file(subfile),
|
2020-04-11 22:36:58 +00:00
|
|
|
subfile.get_basename()
|
|
|
|
)
|
|
|
|
processed_basedir_paths[subfile] = true
|
2020-04-12 22:40:26 +00:00
|
|
|
|
2020-04-11 22:36:58 +00:00
|
|
|
# Iterate over the randomised brush files nyaa
|
|
|
|
for randomised_subdir in randomised_brush_subdirectory_map:
|
|
|
|
if not (randomised_subdir in randomised_brush_subdirectories):
|
|
|
|
var full_paths := []
|
2020-04-12 22:40:26 +00:00
|
|
|
# glue the proper path onto the single file names in the
|
2020-04-11 22:36:58 +00:00
|
|
|
# random brush directory data system, so they can be
|
|
|
|
# opened nya
|
|
|
|
for non_extended_path in randomised_brush_subdirectory_map[randomised_subdir]:
|
|
|
|
full_paths.append(current_main_directory.plus_file(
|
|
|
|
randomised_subdir
|
|
|
|
).plus_file(
|
|
|
|
non_extended_path
|
|
|
|
))
|
|
|
|
# Now load!
|
|
|
|
add_randomised_brush(full_paths, randomised_subdir)
|
|
|
|
# and mark that we are done in the overall map ^.^
|
|
|
|
randomised_brush_subdirectories[randomised_subdir] = true
|
|
|
|
# Now to iterate over the nonrandom brush files inside directories
|
|
|
|
for nonrandomised_subdir in nonrandomised_brush_subdirectory_map:
|
|
|
|
# initialise the set-map for this one if not already present :)
|
2020-04-12 22:40:26 +00:00
|
|
|
if not (nonrandomised_subdir in processed_subdir_paths):
|
2020-04-11 22:36:58 +00:00
|
|
|
processed_subdir_paths[nonrandomised_subdir] = {}
|
|
|
|
# Get the paths within this subdirectory to check if they are
|
|
|
|
# processed or not and if not, then process them.
|
|
|
|
var relpaths_of_contained_nonrandom_brushes : Array = nonrandomised_brush_subdirectory_map[nonrandomised_subdir]
|
|
|
|
for relative_path in relpaths_of_contained_nonrandom_brushes:
|
|
|
|
if not (relative_path in processed_subdir_paths[nonrandomised_subdir]):
|
|
|
|
# We are not yet processed
|
|
|
|
var full_path : String = current_main_directory.plus_file(
|
|
|
|
nonrandomised_subdir
|
|
|
|
).plus_file(
|
|
|
|
relative_path
|
|
|
|
)
|
|
|
|
# Add the path with the tooltip including the directory
|
|
|
|
add_plain_brush(full_path, nonrandomised_subdir.plus_file(
|
|
|
|
relative_path
|
|
|
|
).get_basename())
|
|
|
|
# Mark this as a processed relpath
|
|
|
|
processed_subdir_paths[nonrandomised_subdir][relative_path] = true
|
2020-04-12 22:40:26 +00:00
|
|
|
|
2020-04-11 22:36:58 +00:00
|
|
|
|
|
|
|
|
2019-12-26 17:01:08 +00:00
|
|
|
|
|
|
|
func find_brushes(brushes_dir : Directory, path : String) -> Array:
|
|
|
|
var subdirectories := []
|
2019-12-26 19:36:56 +00:00
|
|
|
var found_random_brush := 0
|
2020-02-09 23:23:33 +00:00
|
|
|
path = Global.root_directory.plus_file(path)
|
2019-12-26 17:01:08 +00:00
|
|
|
brushes_dir.open(path)
|
|
|
|
brushes_dir.list_dir_begin(true)
|
|
|
|
var file := brushes_dir.get_next()
|
|
|
|
while file != "":
|
|
|
|
if file.get_extension().to_upper() == "PNG":
|
|
|
|
var image := Image.new()
|
|
|
|
var err := image.load(path.plus_file(file))
|
|
|
|
if err == OK:
|
2019-12-26 19:36:56 +00:00
|
|
|
if "%" in file:
|
|
|
|
if found_random_brush == 0:
|
|
|
|
found_random_brush = Global.file_brush_container.get_child_count()
|
|
|
|
image.convert(Image.FORMAT_RGBA8)
|
|
|
|
Global.custom_brushes.append(image)
|
2020-02-10 22:06:24 +00:00
|
|
|
Global.create_brush_button(image, Global.Brush_Types.RANDOM_FILE, file.trim_suffix(".png"))
|
2019-12-26 19:36:56 +00:00
|
|
|
else:
|
|
|
|
var brush_button = Global.file_brush_container.get_child(found_random_brush)
|
|
|
|
brush_button.random_brushes.append(image)
|
|
|
|
else:
|
|
|
|
image.convert(Image.FORMAT_RGBA8)
|
|
|
|
Global.custom_brushes.append(image)
|
2020-02-10 22:06:24 +00:00
|
|
|
Global.create_brush_button(image, Global.Brush_Types.FILE, file.trim_suffix(".png"))
|
2019-12-26 17:01:08 +00:00
|
|
|
elif file.get_extension() == "": # Probably a directory
|
|
|
|
var subdir := "./%s" % [file]
|
|
|
|
if brushes_dir.dir_exists(subdir): # If it's an actual directory
|
|
|
|
subdirectories.append(subdir)
|
|
|
|
|
|
|
|
file = brushes_dir.get_next()
|
|
|
|
brushes_dir.list_dir_end()
|
|
|
|
return subdirectories
|
|
|
|
|
2019-12-20 20:22:52 +00:00
|
|
|
func import_gpl(path : String) -> Palette:
|
|
|
|
var result : Palette = null
|
|
|
|
var file = File.new()
|
|
|
|
if file.file_exists(path):
|
|
|
|
file.open(path, File.READ)
|
|
|
|
var text = file.get_as_text()
|
|
|
|
var lines = text.split('\n')
|
|
|
|
var line_number := 0
|
|
|
|
var comments := ""
|
|
|
|
for line in lines:
|
|
|
|
# Check if valid Gimp Palette Library file
|
|
|
|
if line_number == 0:
|
|
|
|
if line != "GIMP Palette":
|
|
|
|
break
|
|
|
|
else:
|
|
|
|
result = Palette.new()
|
|
|
|
var name_start = path.find_last('/') + 1
|
|
|
|
var name_end = path.find_last('.')
|
|
|
|
if name_end > name_start:
|
|
|
|
result.name = path.substr(name_start, name_end - name_start)
|
2019-12-22 18:50:37 +00:00
|
|
|
|
2019-12-20 20:22:52 +00:00
|
|
|
# Comments
|
|
|
|
if line.begins_with('#'):
|
|
|
|
comments += line.trim_prefix('#') + '\n'
|
|
|
|
pass
|
|
|
|
elif line_number > 0 && line.length() >= 12:
|
2020-01-07 22:52:25 +00:00
|
|
|
line = line.replace("\t", " ")
|
|
|
|
var color_data : PoolStringArray = line.split(" ", false, 4)
|
|
|
|
var red : float = color_data[0].to_float() / 255.0
|
|
|
|
var green : float = color_data[1].to_float() / 255.0
|
|
|
|
var blue : float = color_data[2].to_float() / 255.0
|
2019-12-20 20:22:52 +00:00
|
|
|
var color = Color(red, green, blue)
|
2020-01-07 22:52:25 +00:00
|
|
|
result.add_color(color, color_data[3])
|
2019-12-20 20:22:52 +00:00
|
|
|
line_number += 1
|
2019-12-22 18:50:37 +00:00
|
|
|
|
2019-12-20 20:22:52 +00:00
|
|
|
if result:
|
|
|
|
result.comments = comments
|
|
|
|
file.close()
|
|
|
|
|
2019-12-22 18:50:37 +00:00
|
|
|
return result
|
2020-03-02 21:37:52 +00:00
|
|
|
|
|
|
|
func import_png_palette(path: String) -> Palette:
|
|
|
|
var result: Palette = null
|
|
|
|
|
|
|
|
var image := Image.new()
|
|
|
|
var err := image.load(path)
|
|
|
|
if err != OK: # An error occured
|
|
|
|
return null
|
|
|
|
|
|
|
|
var height: int = image.get_height()
|
|
|
|
var width: int = image.get_width()
|
|
|
|
|
|
|
|
result = Palette.new()
|
|
|
|
|
|
|
|
# Iterate all pixels and store unique colors to palete
|
|
|
|
image.lock()
|
|
|
|
for y in range(0, height):
|
|
|
|
for x in range(0, width):
|
|
|
|
var color: Color = image.get_pixel(x, y)
|
|
|
|
if not result.has_color(color):
|
|
|
|
result.add_color(color, "#" + color.to_html())
|
|
|
|
image.unlock()
|
|
|
|
|
|
|
|
var name_start = path.find_last('/') + 1
|
|
|
|
var name_end = path.find_last('.')
|
|
|
|
if name_end > name_start:
|
|
|
|
result.name = path.substr(name_start, name_end - name_start)
|
|
|
|
|
|
|
|
return result
|
|
|
|
pass
|