class_name DeferedResourceLoader extends Node signal on_progress(value : float) signal on_loaded(result : Dictionary[String, Object]) signal on_error(code : ResourceLoadError, res_id : String) enum ResourceLoadError { INVALID_PARAMS, INVALID_RESOURCE, LOAD_FAILED, INVALID_DIRECTORY } func load_resources(directories : Array[String], get_resource_id : Callable, polling_period : float = 0.1) -> void: var res_map : Dictionary[String, Object] = {} var load_list : Dictionary[String, String] = {} if directories == null: on_error.emit(ResourceLoadError.INVALID_PARAMS, "") return for dir_path in directories: # TODO : Recursive if dir_path == null: on_error.emit(ResourceLoadError.INVALID_PARAMS,dir_path) var correct_path := dir_path if dir_path.ends_with("/") else "%s/" % (dir_path) var res_path := ResourceLoader.list_directory(dir_path) for file_name in res_path: var id : String = get_resource_id.call(file_name) if not(id == null or id == ""): var fullname := correct_path + file_name if ResourceLoader.load_threaded_request(fullname) != OK: on_error.emit(ResourceLoadError.LOAD_FAILED, id) else: res_map[id] = null load_list[id] = fullname var total_item_count := load_list.size() on_progress.emit(0.0) var timer : Timer = Timer.new() add_child(timer) timer.set_autostart(false) timer.set_one_shot(false) timer.timeout.connect(_load_loop.bind(res_map, load_list, float(total_item_count), timer)) timer.start(polling_period) func _load_loop(res_map : Dictionary[String, Object], load_list : Dictionary[String, String], total_count : float, timer : Timer) -> void: for r in res_map: if res_map[r] == null: var fullname := load_list[r] var to_erase = true match ResourceLoader.load_threaded_get_status(fullname): ResourceLoader.THREAD_LOAD_INVALID_RESOURCE: on_error.emit(ResourceLoadError.INVALID_RESOURCE, r) ResourceLoader.THREAD_LOAD_FAILED: on_error.emit(ResourceLoadError.LOAD_FAILED, r) ResourceLoader.THREAD_LOAD_LOADED: res_map[r] = ResourceLoader.load_threaded_get(fullname) ResourceLoader.THREAD_LOAD_IN_PROGRESS: to_erase = false if to_erase: load_list.erase(r) on_progress.emit((total_count - load_list.size()) / total_count) if load_list.is_empty(): timer.timeout.disconnect(_load_loop.bind(res_map, load_list, total_count, timer)) timer.queue_free() on_loaded.emit(res_map)
or share this direct link: