extends Node var current_save_path := "" # Stores a filename of a backup file in user:// until user saves manually var backup_save_path = "" var default_autosave_interval := 5 # Minutes onready var autosave_timer : Timer func _ready() -> void: autosave_timer = Timer.new() autosave_timer.one_shot = false autosave_timer.process_mode = Timer.TIMER_PROCESS_IDLE autosave_timer.connect("timeout", self, "_on_Autosave_timeout") add_child(autosave_timer) set_autosave_interval(default_autosave_interval) toggle_autosave(true) # Gets started from preferences dialog func open_pxo_file(path : String, untitled_backup : bool = false) -> void: var file := File.new() var err := file.open_compressed(path, File.READ, File.COMPRESSION_ZSTD) if err == ERR_FILE_UNRECOGNIZED: err = file.open(path, File.READ) # If the file is not compressed open it raw (pre-v0.7) if err != OK: Global.notification_label("File failed to open") file.close() return var file_version := file.get_line() # Example, "v0.7.10-beta" var file_ver_splitted := file_version.split("-") var file_ver_splitted_numbers := file_ver_splitted[0].split(".") # In the above example, the major version would return "0", # the minor version would return "7", the patch "10" # and the status would return "beta" var file_major_version = int(file_ver_splitted_numbers[0].replace("v", "")) var file_minor_version = int(file_ver_splitted_numbers[1]) var _file_patch_version := 0 var _file_status_version : String if file_ver_splitted_numbers.size() > 2: _file_patch_version = int(file_ver_splitted_numbers[2]) if file_ver_splitted.size() > 1: _file_status_version = file_ver_splitted[1] if file_major_version == 0 and file_minor_version < 5: Global.notification_label("File is from an older version of Pixelorama, as such it might not work properly") var frame := 0 Global.layers.clear() var linked_cels := [] if file_major_version >= 0 and file_minor_version > 6: var global_layer_line := file.get_line() while global_layer_line == ".": var layer_name := file.get_line() var layer_visibility := file.get_8() var layer_lock := file.get_8() var layer_new_cels_linked := file.get_8() linked_cels.append(file.get_var()) # Store [Layer name (0), Layer visibility boolean (1), Layer lock boolean (2), Frame container (3), # will new cels be linked boolean (4), Array of linked cels (5)] Global.layers.append([layer_name, layer_visibility, layer_lock, HBoxContainer.new(), layer_new_cels_linked, []]) global_layer_line = file.get_line() var frame_line := file.get_line() Global.clear_canvases() while frame_line == "--": # Load frames var canvas : Canvas = load("res://src/Canvas.tscn").instance() Global.canvas = canvas var width := file.get_16() var height := file.get_16() var layer_i := 0 var layer_line := file.get_line() while layer_line == "-": # Load layers var buffer := file.get_buffer(width * height * 4) if file_major_version == 0 and file_minor_version < 7: var layer_name_old_version = file.get_line() if frame == 0: # Store [Layer name (0), Layer visibility boolean (1), Layer lock boolean (2), Frame container (3), # will new frames be linked boolean (4), Array of linked frames (5)] Global.layers.append([layer_name_old_version, true, false, HBoxContainer.new(), false, []]) var layer_transparency := 1.0 if file_major_version >= 0 and file_minor_version > 5: layer_transparency = file.get_float() var image := Image.new() image.create_from_data(width, height, false, Image.FORMAT_RGBA8, buffer) image.lock() var tex := ImageTexture.new() tex.create_from_image(image, 0) canvas.layers.append([image, tex, layer_transparency]) if file_major_version >= 0 and file_minor_version >= 7: if frame in linked_cels[layer_i]: Global.layers[layer_i][5].append(canvas) layer_i += 1 layer_line = file.get_line() var guide_line := file.get_line() # "guideline" no pun intended while guide_line == "|": # Load guides var guide := Guide.new() guide.default_color = Color.purple guide.type = file.get_8() if guide.type == guide.Types.HORIZONTAL: guide.add_point(Vector2(-99999, file.get_16())) guide.add_point(Vector2(99999, file.get_16())) else: guide.add_point(Vector2(file.get_16(), -99999)) guide.add_point(Vector2(file.get_16(), 99999)) guide.has_focus = false canvas.add_child(guide) guide_line = file.get_line() canvas.size = Vector2(width, height) Global.canvases.append(canvas) canvas.frame = frame Global.canvas_parent.add_child(canvas) frame_line = file.get_line() frame += 1 Global.canvases = Global.canvases # Just to call Global.canvases_changed Global.current_layer = Global.layers.size() - 1 Global.current_frame = frame - 1 Global.layers = Global.layers # Just to call Global.layers_changed # Load tool options Global.color_pickers[0].color = file.get_var() Global.color_pickers[1].color = file.get_var() Global.left_brush_size = file.get_8() Global.brush_size_edits[0].value = Global.left_brush_size Global.right_brush_size = file.get_8() Global.brush_size_edits[1].value = Global.right_brush_size if file_major_version == 0 and file_minor_version < 7: var left_palette = file.get_var() var right_palette = file.get_var() for color in left_palette: Global.color_pickers[0].get_picker().add_preset(color) for color in right_palette: Global.color_pickers[1].get_picker().add_preset(color) # Load custom brushes Global.custom_brushes.resize(Global.brushes_from_files) Global.remove_brush_buttons() var brush_line := file.get_line() while brush_line == "/": var b_width := file.get_16() var b_height := file.get_16() var buffer := file.get_buffer(b_width * b_height * 4) var image := Image.new() image.create_from_data(b_width, b_height, false, Image.FORMAT_RGBA8, buffer) Global.custom_brushes.append(image) Global.create_brush_button(image) brush_line = file.get_line() if file_major_version >= 0 and file_minor_version > 6: var tag_line := file.get_line() while tag_line == ".T/": var tag_name := file.get_line() var tag_color : Color = file.get_var() var tag_from := file.get_8() var tag_to := file.get_8() Global.animation_tags.append([tag_name, tag_color, tag_from, tag_to]) Global.animation_tags = Global.animation_tags # To execute animation_tags_changed() tag_line = file.get_line() file.close() if not untitled_backup: # Untitled backup should not change window title and save path current_save_path = path Global.window_title = path.get_file() + " - Pixelorama " + Global.current_version Global.project_has_changed = false func save_pxo_file(path : String, autosave : bool) -> void: var file := File.new() var err := file.open_compressed(path, File.WRITE, File.COMPRESSION_ZSTD) if err == OK: # Store Pixelorama version file.store_line(Global.current_version) # Store Global layers for layer in Global.layers: file.store_line(".") file.store_line(layer[0]) # Layer name file.store_8(layer[1]) # Layer visibility file.store_8(layer[2]) # Layer lock file.store_8(layer[4]) # Future cels linked var linked_cels := [] for canvas in layer[5]: linked_cels.append(canvas.frame) file.store_var(linked_cels) # Linked cels as cel numbers file.store_line("END_GLOBAL_LAYERS") # Store frames for canvas in Global.canvases: file.store_line("--") file.store_16(canvas.size.x) file.store_16(canvas.size.y) for layer in canvas.layers: # Store canvas layers file.store_line("-") file.store_buffer(layer[0].get_data()) file.store_float(layer[2]) # Layer transparency file.store_line("END_LAYERS") # Store guides for child in canvas.get_children(): if child is Guide: file.store_line("|") file.store_8(child.type) if child.type == child.Types.HORIZONTAL: file.store_16(child.points[0].y) file.store_16(child.points[1].y) else: file.store_16(child.points[1].x) file.store_16(child.points[0].x) file.store_line("END_GUIDES") file.store_line("END_FRAMES") # Save tool options var left_color : Color = Global.color_pickers[0].color var right_color : Color = Global.color_pickers[1].color var left_brush_size : int = Global.left_brush_size var right_brush_size : int = Global.right_brush_size file.store_var(left_color) file.store_var(right_color) file.store_8(left_brush_size) file.store_8(right_brush_size) # Save custom brushes for i in range(Global.brushes_from_files, Global.custom_brushes.size()): var brush = Global.custom_brushes[i] file.store_line("/") file.store_16(brush.get_size().x) file.store_16(brush.get_size().y) file.store_buffer(brush.get_data()) file.store_line("END_BRUSHES") # Store animation tags for tag in Global.animation_tags: file.store_line(".T/") file.store_line(tag[0]) # Tag name file.store_var(tag[1]) # Tag color file.store_8(tag[2]) # Tag "from", the first frame file.store_8(tag[3]) # Tag "to", the last frame file.store_line("END_FRAME_TAGS") file.close() if Global.project_has_changed and not autosave: Global.project_has_changed = false if autosave: Global.notification_label("File autosaved") else: # First remove backup then set current save path remove_backup() current_save_path = path Global.notification_label("File saved") if backup_save_path == "": Global.window_title = path.get_file() + " - Pixelorama " + Global.current_version else: Global.notification_label("File failed to save") func toggle_autosave(enable : bool) -> void: if enable: autosave_timer.start() else: autosave_timer.stop() func set_autosave_interval(interval : float) -> void: autosave_timer.wait_time = interval * 60 # Interval parameter is in minutes, wait_time is seconds autosave_timer.start() func _on_Autosave_timeout() -> void: if backup_save_path == "": # Create a new backup file if it doesn't exist yet backup_save_path = "user://backup-" + String(OS.get_unix_time()) store_backup_path() save_pxo_file(backup_save_path, true) # Backup paths are stored in two ways: # 1) User already manually saved and defined a save path -> {current_save_path, backup_save_path} # 2) User didn't manually saved, "untitled" backup is stored -> {backup_save_path, backup_save_path} func store_backup_path() -> void: if current_save_path != "": # Remove "untitled" backup if it existed on this project instance if Global.config_cache.has_section_key("backups", backup_save_path): Global.config_cache.erase_section_key("backups", backup_save_path) Global.config_cache.set_value("backups", current_save_path, backup_save_path) else: Global.config_cache.set_value("backups", backup_save_path, backup_save_path) Global.config_cache.save("user://cache.ini") func remove_backup() -> void: # Remove backup file if backup_save_path != "": if current_save_path != "": remove_backup_by_path(current_save_path, backup_save_path) else: # If manual save was not yet done - remove "untitled" backup remove_backup_by_path(backup_save_path, backup_save_path) backup_save_path = "" func remove_backup_by_path(project_path : String, backup_path : String) -> void: Directory.new().remove(backup_path) Global.config_cache.erase_section_key("backups", project_path) Global.config_cache.save("user://cache.ini") func reload_backup_file(project_path : String, backup_path : String) -> void: # If project path is the same as backup save path -> the backup was untitled open_pxo_file(backup_path, project_path == backup_path) backup_save_path = backup_path if project_path != backup_path: current_save_path = project_path Global.window_title = project_path.get_file() + " - Pixelorama(*) " + Global.current_version Global.project_has_changed = true Global.notification_label("Backup reloaded")