Implement a basic extension system
Importing .pck or .zip Godot resource pack files into Pixelorama is now possible. This needs to be documented properly, but here's the basic idea, for now at least. This is super early work and I haven't tested it with a proper extension yet, so all of this could be a subject of change. I tested it with a custom theme extension though and it seems to be working perfectly.
Importing resource pack files, either by dragging and dropping them into the app window or by going to Edit>Preferences>Extensions>Add Extension, copies the files into user://extensions/. Extensions can be enabled/disabled and uninstalled. Uninstalling them deletes the resource pack files from user://extensions/.
The extension project source files need to be in a folder inside src/Extensions/ with the same name as the .pck or .zip file. **This is required for now, otherwise it will not work.** Inside that folder there also needs to be an extension.json file, with a structure similar to this:
{
"name": "ExtensionName",
"display_name": "Extension Name",
"description": "A Pixelorama extension",
"author": "Orama Interactive",
"version": "0.1",
"license": "MIT",
"nodes": [
"ExtensionExample.tscn"
]
}
The `nodes` array leads to the packed scene files with the nodes that are to be instantiated. **The root nodes of these scenes need to have the same name as the .tscn files they belong to.** The scripts of these nodes should have _enter_tree() and _exit_tree() methods to handle the extension enabling/disabling (or even uninstalling) logic. Note that .json files need to be included in the export options while exporting the extension from Godot.
Enabling an extension means that the scenes found in the extension.json's "nodes" array get instantiated, and disabling gets rid of these nodes from Pixelorama's SceneTree.
2022-02-19 01:21:08 +00:00
|
|
|
extends Control
|
|
|
|
|
|
|
|
const EXTENSIONS_PATH := "user://extensions"
|
|
|
|
|
|
|
|
var extensions := {} # Extension name : Extension class
|
|
|
|
var extension_selected := -1
|
2022-07-20 12:22:43 +00:00
|
|
|
var damaged_extension: String
|
Implement a basic extension system
Importing .pck or .zip Godot resource pack files into Pixelorama is now possible. This needs to be documented properly, but here's the basic idea, for now at least. This is super early work and I haven't tested it with a proper extension yet, so all of this could be a subject of change. I tested it with a custom theme extension though and it seems to be working perfectly.
Importing resource pack files, either by dragging and dropping them into the app window or by going to Edit>Preferences>Extensions>Add Extension, copies the files into user://extensions/. Extensions can be enabled/disabled and uninstalled. Uninstalling them deletes the resource pack files from user://extensions/.
The extension project source files need to be in a folder inside src/Extensions/ with the same name as the .pck or .zip file. **This is required for now, otherwise it will not work.** Inside that folder there also needs to be an extension.json file, with a structure similar to this:
{
"name": "ExtensionName",
"display_name": "Extension Name",
"description": "A Pixelorama extension",
"author": "Orama Interactive",
"version": "0.1",
"license": "MIT",
"nodes": [
"ExtensionExample.tscn"
]
}
The `nodes` array leads to the packed scene files with the nodes that are to be instantiated. **The root nodes of these scenes need to have the same name as the .tscn files they belong to.** The scripts of these nodes should have _enter_tree() and _exit_tree() methods to handle the extension enabling/disabling (or even uninstalling) logic. Note that .json files need to be included in the export options while exporting the extension from Godot.
Enabling an extension means that the scenes found in the extension.json's "nodes" array get instantiated, and disabling gets rid of these nodes from Pixelorama's SceneTree.
2022-02-19 01:21:08 +00:00
|
|
|
|
|
|
|
onready var extension_list: ItemList = $InstalledExtensions
|
|
|
|
onready var enable_button: Button = $HBoxContainer/EnableButton
|
|
|
|
onready var uninstall_button: Button = $HBoxContainer/UninstallButton
|
|
|
|
onready var extension_parent: Node = Global.control.get_node("Extensions")
|
|
|
|
|
|
|
|
|
|
|
|
class Extension:
|
|
|
|
var file_name := ""
|
|
|
|
var display_name := ""
|
|
|
|
var description := ""
|
|
|
|
var author := ""
|
|
|
|
var version := ""
|
|
|
|
var license := ""
|
|
|
|
var nodes := []
|
|
|
|
var enabled := false
|
|
|
|
|
|
|
|
func serialize(dict: Dictionary) -> void:
|
|
|
|
if dict.has("name"):
|
|
|
|
file_name = dict["name"]
|
|
|
|
if dict.has("display_name"):
|
|
|
|
display_name = dict["display_name"]
|
|
|
|
if dict.has("description"):
|
|
|
|
description = dict["description"]
|
|
|
|
if dict.has("author"):
|
|
|
|
author = dict["author"]
|
|
|
|
if dict.has("version"):
|
|
|
|
version = dict["version"]
|
|
|
|
if dict.has("license"):
|
|
|
|
license = dict["license"]
|
|
|
|
if dict.has("nodes"):
|
|
|
|
nodes = dict["nodes"]
|
|
|
|
|
|
|
|
|
|
|
|
func _ready() -> void:
|
|
|
|
if OS.get_name() == "HTML5":
|
|
|
|
$HBoxContainer/AddExtensionButton.disabled = true
|
|
|
|
$HBoxContainer/OpenFolderButton.visible = false
|
|
|
|
|
|
|
|
var dir := Directory.new()
|
|
|
|
var file_names := [] # Array of String(s)
|
|
|
|
dir.make_dir(EXTENSIONS_PATH)
|
|
|
|
if dir.open(EXTENSIONS_PATH) == OK:
|
|
|
|
dir.list_dir_begin()
|
|
|
|
var file_name = dir.get_next()
|
|
|
|
while file_name != "":
|
|
|
|
var ext: String = file_name.to_lower().get_extension()
|
|
|
|
if !dir.current_is_dir() and ext in ["pck", "zip"]:
|
|
|
|
file_names.append(file_name)
|
|
|
|
file_name = dir.get_next()
|
|
|
|
|
|
|
|
if file_names.empty():
|
|
|
|
return
|
|
|
|
|
|
|
|
for file_name in file_names:
|
|
|
|
_add_extension(file_name)
|
|
|
|
|
|
|
|
|
|
|
|
func install_extension(path: String) -> void:
|
|
|
|
var dir := Directory.new()
|
|
|
|
var file_name: String = path.get_file()
|
|
|
|
dir.copy(path, EXTENSIONS_PATH.plus_file(file_name))
|
|
|
|
_add_extension(file_name)
|
|
|
|
|
|
|
|
|
2022-07-05 21:12:38 +00:00
|
|
|
func _uninstall_extension(file_name := "", remove_file := true, item := extension_selected) -> void:
|
|
|
|
if remove_file:
|
|
|
|
var dir := Directory.new()
|
|
|
|
var err := dir.remove(EXTENSIONS_PATH.plus_file(file_name))
|
|
|
|
if err != OK:
|
|
|
|
print(err)
|
|
|
|
return
|
|
|
|
|
|
|
|
var extension: Extension = extensions[file_name]
|
|
|
|
extension.enabled = false
|
|
|
|
_enable_extension(extension, false)
|
|
|
|
|
|
|
|
extensions.erase(file_name)
|
|
|
|
extension_list.remove_item(item)
|
|
|
|
extension_selected = -1
|
|
|
|
enable_button.disabled = true
|
|
|
|
uninstall_button.disabled = true
|
|
|
|
|
|
|
|
|
Implement a basic extension system
Importing .pck or .zip Godot resource pack files into Pixelorama is now possible. This needs to be documented properly, but here's the basic idea, for now at least. This is super early work and I haven't tested it with a proper extension yet, so all of this could be a subject of change. I tested it with a custom theme extension though and it seems to be working perfectly.
Importing resource pack files, either by dragging and dropping them into the app window or by going to Edit>Preferences>Extensions>Add Extension, copies the files into user://extensions/. Extensions can be enabled/disabled and uninstalled. Uninstalling them deletes the resource pack files from user://extensions/.
The extension project source files need to be in a folder inside src/Extensions/ with the same name as the .pck or .zip file. **This is required for now, otherwise it will not work.** Inside that folder there also needs to be an extension.json file, with a structure similar to this:
{
"name": "ExtensionName",
"display_name": "Extension Name",
"description": "A Pixelorama extension",
"author": "Orama Interactive",
"version": "0.1",
"license": "MIT",
"nodes": [
"ExtensionExample.tscn"
]
}
The `nodes` array leads to the packed scene files with the nodes that are to be instantiated. **The root nodes of these scenes need to have the same name as the .tscn files they belong to.** The scripts of these nodes should have _enter_tree() and _exit_tree() methods to handle the extension enabling/disabling (or even uninstalling) logic. Note that .json files need to be included in the export options while exporting the extension from Godot.
Enabling an extension means that the scenes found in the extension.json's "nodes" array get instantiated, and disabling gets rid of these nodes from Pixelorama's SceneTree.
2022-02-19 01:21:08 +00:00
|
|
|
func _add_extension(file_name: String) -> void:
|
2022-07-20 12:22:43 +00:00
|
|
|
var tester_file := File.new() # For testing and deleting damaged extensions
|
|
|
|
var remover_directory := Directory.new()
|
|
|
|
# Remove any extension that was proven guilty before this extension is loaded
|
|
|
|
if tester_file.file_exists(EXTENSIONS_PATH.plus_file("Faulty.txt")):
|
|
|
|
# This code will only run if pixelorama crashed
|
|
|
|
var faulty_path = EXTENSIONS_PATH.plus_file("Faulty.txt")
|
|
|
|
tester_file.open(faulty_path, File.READ)
|
|
|
|
damaged_extension = tester_file.get_as_text()
|
|
|
|
tester_file.close()
|
|
|
|
remover_directory.remove(EXTENSIONS_PATH.plus_file(damaged_extension))
|
|
|
|
remover_directory.remove(EXTENSIONS_PATH.plus_file("Faulty.txt"))
|
|
|
|
|
|
|
|
# Don't load a deleted extension
|
|
|
|
if damaged_extension == file_name:
|
|
|
|
# This code will only run if pixelorama crashed
|
|
|
|
damaged_extension = ""
|
|
|
|
return
|
|
|
|
|
|
|
|
# The new (about to load) extension will be considered guilty till it's proven innocent
|
|
|
|
tester_file.open(EXTENSIONS_PATH.plus_file("Faulty.txt"), File.WRITE)
|
|
|
|
tester_file.store_string(file_name) # Guilty till proven innocent ;)
|
|
|
|
tester_file.close()
|
|
|
|
|
Implement a basic extension system
Importing .pck or .zip Godot resource pack files into Pixelorama is now possible. This needs to be documented properly, but here's the basic idea, for now at least. This is super early work and I haven't tested it with a proper extension yet, so all of this could be a subject of change. I tested it with a custom theme extension though and it seems to be working perfectly.
Importing resource pack files, either by dragging and dropping them into the app window or by going to Edit>Preferences>Extensions>Add Extension, copies the files into user://extensions/. Extensions can be enabled/disabled and uninstalled. Uninstalling them deletes the resource pack files from user://extensions/.
The extension project source files need to be in a folder inside src/Extensions/ with the same name as the .pck or .zip file. **This is required for now, otherwise it will not work.** Inside that folder there also needs to be an extension.json file, with a structure similar to this:
{
"name": "ExtensionName",
"display_name": "Extension Name",
"description": "A Pixelorama extension",
"author": "Orama Interactive",
"version": "0.1",
"license": "MIT",
"nodes": [
"ExtensionExample.tscn"
]
}
The `nodes` array leads to the packed scene files with the nodes that are to be instantiated. **The root nodes of these scenes need to have the same name as the .tscn files they belong to.** The scripts of these nodes should have _enter_tree() and _exit_tree() methods to handle the extension enabling/disabling (or even uninstalling) logic. Note that .json files need to be included in the export options while exporting the extension from Godot.
Enabling an extension means that the scenes found in the extension.json's "nodes" array get instantiated, and disabling gets rid of these nodes from Pixelorama's SceneTree.
2022-02-19 01:21:08 +00:00
|
|
|
if extensions.has(file_name):
|
2022-07-05 21:12:38 +00:00
|
|
|
var item := -1
|
|
|
|
for i in extension_list.get_item_count():
|
|
|
|
if extension_list.get_item_metadata(i) == file_name:
|
|
|
|
item = i
|
|
|
|
break
|
|
|
|
if item == -1:
|
|
|
|
print("Failed to find %s" % file_name)
|
|
|
|
return
|
|
|
|
_uninstall_extension(file_name, false, item)
|
|
|
|
# Wait two frames so the previous nodes can get freed
|
|
|
|
yield(get_tree(), "idle_frame")
|
|
|
|
yield(get_tree(), "idle_frame")
|
|
|
|
|
Implement a basic extension system
Importing .pck or .zip Godot resource pack files into Pixelorama is now possible. This needs to be documented properly, but here's the basic idea, for now at least. This is super early work and I haven't tested it with a proper extension yet, so all of this could be a subject of change. I tested it with a custom theme extension though and it seems to be working perfectly.
Importing resource pack files, either by dragging and dropping them into the app window or by going to Edit>Preferences>Extensions>Add Extension, copies the files into user://extensions/. Extensions can be enabled/disabled and uninstalled. Uninstalling them deletes the resource pack files from user://extensions/.
The extension project source files need to be in a folder inside src/Extensions/ with the same name as the .pck or .zip file. **This is required for now, otherwise it will not work.** Inside that folder there also needs to be an extension.json file, with a structure similar to this:
{
"name": "ExtensionName",
"display_name": "Extension Name",
"description": "A Pixelorama extension",
"author": "Orama Interactive",
"version": "0.1",
"license": "MIT",
"nodes": [
"ExtensionExample.tscn"
]
}
The `nodes` array leads to the packed scene files with the nodes that are to be instantiated. **The root nodes of these scenes need to have the same name as the .tscn files they belong to.** The scripts of these nodes should have _enter_tree() and _exit_tree() methods to handle the extension enabling/disabling (or even uninstalling) logic. Note that .json files need to be included in the export options while exporting the extension from Godot.
Enabling an extension means that the scenes found in the extension.json's "nodes" array get instantiated, and disabling gets rid of these nodes from Pixelorama's SceneTree.
2022-02-19 01:21:08 +00:00
|
|
|
var file_name_no_ext: String = file_name.get_basename()
|
|
|
|
var file_path: String = EXTENSIONS_PATH.plus_file(file_name)
|
|
|
|
var success := ProjectSettings.load_resource_pack(file_path)
|
|
|
|
if !success:
|
|
|
|
print("Failed loading resource pack.")
|
|
|
|
var dir := Directory.new()
|
|
|
|
dir.remove(file_path)
|
|
|
|
return
|
|
|
|
|
|
|
|
var extension_path: String = "res://src/Extensions/%s/" % file_name_no_ext
|
|
|
|
var extension_config_file_path: String = extension_path.plus_file("extension.json")
|
|
|
|
var extension_config_file := File.new()
|
|
|
|
var err := extension_config_file.open(extension_config_file_path, File.READ)
|
|
|
|
if err != OK:
|
|
|
|
print("Error loading config file: ", err)
|
|
|
|
extension_config_file.close()
|
|
|
|
return
|
|
|
|
|
|
|
|
var extension_json = parse_json(extension_config_file.get_as_text())
|
|
|
|
extension_config_file.close()
|
|
|
|
|
|
|
|
if !extension_json:
|
|
|
|
print("No JSON data found.")
|
|
|
|
return
|
|
|
|
|
2023-01-08 17:17:42 +00:00
|
|
|
if extension_json.has("supported_api_versions"):
|
|
|
|
var supported_api_versions = extension_json["supported_api_versions"]
|
|
|
|
if typeof(supported_api_versions) == TYPE_ARRAY:
|
|
|
|
if !ExtensionsApi.get_api_version() in supported_api_versions:
|
|
|
|
var err_text = (
|
|
|
|
"The extension %s will not work on this version of Pixelorama \n"
|
|
|
|
% file_name_no_ext
|
|
|
|
)
|
|
|
|
var required_text = "Requires Api : %s" % str(supported_api_versions)
|
|
|
|
Global.error_dialog.set_text(str(err_text, required_text))
|
|
|
|
Global.error_dialog.popup_centered()
|
|
|
|
Global.dialog_open(true)
|
|
|
|
print("Incompatible API")
|
2023-01-10 17:26:13 +00:00
|
|
|
# Don't put it in faulty, (it's merely incompatible)
|
|
|
|
remover_directory.remove(EXTENSIONS_PATH.plus_file("Faulty.txt"))
|
2023-01-08 17:17:42 +00:00
|
|
|
return
|
|
|
|
|
Implement a basic extension system
Importing .pck or .zip Godot resource pack files into Pixelorama is now possible. This needs to be documented properly, but here's the basic idea, for now at least. This is super early work and I haven't tested it with a proper extension yet, so all of this could be a subject of change. I tested it with a custom theme extension though and it seems to be working perfectly.
Importing resource pack files, either by dragging and dropping them into the app window or by going to Edit>Preferences>Extensions>Add Extension, copies the files into user://extensions/. Extensions can be enabled/disabled and uninstalled. Uninstalling them deletes the resource pack files from user://extensions/.
The extension project source files need to be in a folder inside src/Extensions/ with the same name as the .pck or .zip file. **This is required for now, otherwise it will not work.** Inside that folder there also needs to be an extension.json file, with a structure similar to this:
{
"name": "ExtensionName",
"display_name": "Extension Name",
"description": "A Pixelorama extension",
"author": "Orama Interactive",
"version": "0.1",
"license": "MIT",
"nodes": [
"ExtensionExample.tscn"
]
}
The `nodes` array leads to the packed scene files with the nodes that are to be instantiated. **The root nodes of these scenes need to have the same name as the .tscn files they belong to.** The scripts of these nodes should have _enter_tree() and _exit_tree() methods to handle the extension enabling/disabling (or even uninstalling) logic. Note that .json files need to be included in the export options while exporting the extension from Godot.
Enabling an extension means that the scenes found in the extension.json's "nodes" array get instantiated, and disabling gets rid of these nodes from Pixelorama's SceneTree.
2022-02-19 01:21:08 +00:00
|
|
|
var extension := Extension.new()
|
|
|
|
extension.serialize(extension_json)
|
|
|
|
extensions[file_name] = extension
|
|
|
|
extension_list.add_item(extension.display_name)
|
|
|
|
var item_count: int = extension_list.get_item_count() - 1
|
|
|
|
extension_list.set_item_tooltip(item_count, extension.description)
|
|
|
|
extension_list.set_item_metadata(item_count, file_name)
|
|
|
|
extension.enabled = Global.config_cache.get_value("extensions", extension.file_name, false)
|
|
|
|
if extension.enabled:
|
|
|
|
_enable_extension(extension)
|
|
|
|
|
2022-07-20 12:22:43 +00:00
|
|
|
# If an extension doesn't crash pixelorama then it is proven innocent
|
2023-06-15 15:00:29 +00:00
|
|
|
# And we should now delete its "Faulty.txt" file
|
2022-07-20 12:22:43 +00:00
|
|
|
remover_directory.remove(EXTENSIONS_PATH.plus_file("Faulty.txt"))
|
|
|
|
|
Implement a basic extension system
Importing .pck or .zip Godot resource pack files into Pixelorama is now possible. This needs to be documented properly, but here's the basic idea, for now at least. This is super early work and I haven't tested it with a proper extension yet, so all of this could be a subject of change. I tested it with a custom theme extension though and it seems to be working perfectly.
Importing resource pack files, either by dragging and dropping them into the app window or by going to Edit>Preferences>Extensions>Add Extension, copies the files into user://extensions/. Extensions can be enabled/disabled and uninstalled. Uninstalling them deletes the resource pack files from user://extensions/.
The extension project source files need to be in a folder inside src/Extensions/ with the same name as the .pck or .zip file. **This is required for now, otherwise it will not work.** Inside that folder there also needs to be an extension.json file, with a structure similar to this:
{
"name": "ExtensionName",
"display_name": "Extension Name",
"description": "A Pixelorama extension",
"author": "Orama Interactive",
"version": "0.1",
"license": "MIT",
"nodes": [
"ExtensionExample.tscn"
]
}
The `nodes` array leads to the packed scene files with the nodes that are to be instantiated. **The root nodes of these scenes need to have the same name as the .tscn files they belong to.** The scripts of these nodes should have _enter_tree() and _exit_tree() methods to handle the extension enabling/disabling (or even uninstalling) logic. Note that .json files need to be included in the export options while exporting the extension from Godot.
Enabling an extension means that the scenes found in the extension.json's "nodes" array get instantiated, and disabling gets rid of these nodes from Pixelorama's SceneTree.
2022-02-19 01:21:08 +00:00
|
|
|
|
|
|
|
func _enable_extension(extension: Extension, save_to_config := true) -> void:
|
|
|
|
var extension_path: String = "res://src/Extensions/%s/" % extension.file_name
|
2022-02-26 14:14:13 +00:00
|
|
|
|
|
|
|
# A unique id for the extension (currently set to file_name). More parameters (version etc.)
|
|
|
|
# can be easily added using the str() function. for example
|
|
|
|
# var id: String = str(extension.file_name, extension.version)
|
|
|
|
var id: String = extension.file_name
|
|
|
|
|
Implement a basic extension system
Importing .pck or .zip Godot resource pack files into Pixelorama is now possible. This needs to be documented properly, but here's the basic idea, for now at least. This is super early work and I haven't tested it with a proper extension yet, so all of this could be a subject of change. I tested it with a custom theme extension though and it seems to be working perfectly.
Importing resource pack files, either by dragging and dropping them into the app window or by going to Edit>Preferences>Extensions>Add Extension, copies the files into user://extensions/. Extensions can be enabled/disabled and uninstalled. Uninstalling them deletes the resource pack files from user://extensions/.
The extension project source files need to be in a folder inside src/Extensions/ with the same name as the .pck or .zip file. **This is required for now, otherwise it will not work.** Inside that folder there also needs to be an extension.json file, with a structure similar to this:
{
"name": "ExtensionName",
"display_name": "Extension Name",
"description": "A Pixelorama extension",
"author": "Orama Interactive",
"version": "0.1",
"license": "MIT",
"nodes": [
"ExtensionExample.tscn"
]
}
The `nodes` array leads to the packed scene files with the nodes that are to be instantiated. **The root nodes of these scenes need to have the same name as the .tscn files they belong to.** The scripts of these nodes should have _enter_tree() and _exit_tree() methods to handle the extension enabling/disabling (or even uninstalling) logic. Note that .json files need to be included in the export options while exporting the extension from Godot.
Enabling an extension means that the scenes found in the extension.json's "nodes" array get instantiated, and disabling gets rid of these nodes from Pixelorama's SceneTree.
2022-02-19 01:21:08 +00:00
|
|
|
if extension.enabled:
|
2023-01-10 17:26:13 +00:00
|
|
|
ExtensionsApi.clear_history(extension.file_name)
|
Implement a basic extension system
Importing .pck or .zip Godot resource pack files into Pixelorama is now possible. This needs to be documented properly, but here's the basic idea, for now at least. This is super early work and I haven't tested it with a proper extension yet, so all of this could be a subject of change. I tested it with a custom theme extension though and it seems to be working perfectly.
Importing resource pack files, either by dragging and dropping them into the app window or by going to Edit>Preferences>Extensions>Add Extension, copies the files into user://extensions/. Extensions can be enabled/disabled and uninstalled. Uninstalling them deletes the resource pack files from user://extensions/.
The extension project source files need to be in a folder inside src/Extensions/ with the same name as the .pck or .zip file. **This is required for now, otherwise it will not work.** Inside that folder there also needs to be an extension.json file, with a structure similar to this:
{
"name": "ExtensionName",
"display_name": "Extension Name",
"description": "A Pixelorama extension",
"author": "Orama Interactive",
"version": "0.1",
"license": "MIT",
"nodes": [
"ExtensionExample.tscn"
]
}
The `nodes` array leads to the packed scene files with the nodes that are to be instantiated. **The root nodes of these scenes need to have the same name as the .tscn files they belong to.** The scripts of these nodes should have _enter_tree() and _exit_tree() methods to handle the extension enabling/disabling (or even uninstalling) logic. Note that .json files need to be included in the export options while exporting the extension from Godot.
Enabling an extension means that the scenes found in the extension.json's "nodes" array get instantiated, and disabling gets rid of these nodes from Pixelorama's SceneTree.
2022-02-19 01:21:08 +00:00
|
|
|
for node in extension.nodes:
|
|
|
|
var scene_path: String = extension_path.plus_file(node)
|
|
|
|
var extension_scene: PackedScene = load(scene_path)
|
|
|
|
if extension_scene:
|
|
|
|
var extension_node: Node = extension_scene.instance()
|
|
|
|
extension_parent.add_child(extension_node)
|
2022-02-26 14:14:13 +00:00
|
|
|
extension_node.add_to_group(id) # Keep track of what to remove later
|
2022-07-05 21:12:38 +00:00
|
|
|
else:
|
|
|
|
print("Failed to load extension %s" % id)
|
Implement a basic extension system
Importing .pck or .zip Godot resource pack files into Pixelorama is now possible. This needs to be documented properly, but here's the basic idea, for now at least. This is super early work and I haven't tested it with a proper extension yet, so all of this could be a subject of change. I tested it with a custom theme extension though and it seems to be working perfectly.
Importing resource pack files, either by dragging and dropping them into the app window or by going to Edit>Preferences>Extensions>Add Extension, copies the files into user://extensions/. Extensions can be enabled/disabled and uninstalled. Uninstalling them deletes the resource pack files from user://extensions/.
The extension project source files need to be in a folder inside src/Extensions/ with the same name as the .pck or .zip file. **This is required for now, otherwise it will not work.** Inside that folder there also needs to be an extension.json file, with a structure similar to this:
{
"name": "ExtensionName",
"display_name": "Extension Name",
"description": "A Pixelorama extension",
"author": "Orama Interactive",
"version": "0.1",
"license": "MIT",
"nodes": [
"ExtensionExample.tscn"
]
}
The `nodes` array leads to the packed scene files with the nodes that are to be instantiated. **The root nodes of these scenes need to have the same name as the .tscn files they belong to.** The scripts of these nodes should have _enter_tree() and _exit_tree() methods to handle the extension enabling/disabling (or even uninstalling) logic. Note that .json files need to be included in the export options while exporting the extension from Godot.
Enabling an extension means that the scenes found in the extension.json's "nodes" array get instantiated, and disabling gets rid of these nodes from Pixelorama's SceneTree.
2022-02-19 01:21:08 +00:00
|
|
|
else:
|
2022-02-26 14:14:13 +00:00
|
|
|
for ext_node in extension_parent.get_children():
|
2023-02-02 01:06:23 +00:00
|
|
|
if ext_node.is_in_group(id): # Node for extension found
|
Implement a basic extension system
Importing .pck or .zip Godot resource pack files into Pixelorama is now possible. This needs to be documented properly, but here's the basic idea, for now at least. This is super early work and I haven't tested it with a proper extension yet, so all of this could be a subject of change. I tested it with a custom theme extension though and it seems to be working perfectly.
Importing resource pack files, either by dragging and dropping them into the app window or by going to Edit>Preferences>Extensions>Add Extension, copies the files into user://extensions/. Extensions can be enabled/disabled and uninstalled. Uninstalling them deletes the resource pack files from user://extensions/.
The extension project source files need to be in a folder inside src/Extensions/ with the same name as the .pck or .zip file. **This is required for now, otherwise it will not work.** Inside that folder there also needs to be an extension.json file, with a structure similar to this:
{
"name": "ExtensionName",
"display_name": "Extension Name",
"description": "A Pixelorama extension",
"author": "Orama Interactive",
"version": "0.1",
"license": "MIT",
"nodes": [
"ExtensionExample.tscn"
]
}
The `nodes` array leads to the packed scene files with the nodes that are to be instantiated. **The root nodes of these scenes need to have the same name as the .tscn files they belong to.** The scripts of these nodes should have _enter_tree() and _exit_tree() methods to handle the extension enabling/disabling (or even uninstalling) logic. Note that .json files need to be included in the export options while exporting the extension from Godot.
Enabling an extension means that the scenes found in the extension.json's "nodes" array get instantiated, and disabling gets rid of these nodes from Pixelorama's SceneTree.
2022-02-19 01:21:08 +00:00
|
|
|
extension_parent.remove_child(ext_node)
|
|
|
|
ext_node.queue_free()
|
2023-01-10 17:26:13 +00:00
|
|
|
ExtensionsApi.check_sanity(extension.file_name)
|
Implement a basic extension system
Importing .pck or .zip Godot resource pack files into Pixelorama is now possible. This needs to be documented properly, but here's the basic idea, for now at least. This is super early work and I haven't tested it with a proper extension yet, so all of this could be a subject of change. I tested it with a custom theme extension though and it seems to be working perfectly.
Importing resource pack files, either by dragging and dropping them into the app window or by going to Edit>Preferences>Extensions>Add Extension, copies the files into user://extensions/. Extensions can be enabled/disabled and uninstalled. Uninstalling them deletes the resource pack files from user://extensions/.
The extension project source files need to be in a folder inside src/Extensions/ with the same name as the .pck or .zip file. **This is required for now, otherwise it will not work.** Inside that folder there also needs to be an extension.json file, with a structure similar to this:
{
"name": "ExtensionName",
"display_name": "Extension Name",
"description": "A Pixelorama extension",
"author": "Orama Interactive",
"version": "0.1",
"license": "MIT",
"nodes": [
"ExtensionExample.tscn"
]
}
The `nodes` array leads to the packed scene files with the nodes that are to be instantiated. **The root nodes of these scenes need to have the same name as the .tscn files they belong to.** The scripts of these nodes should have _enter_tree() and _exit_tree() methods to handle the extension enabling/disabling (or even uninstalling) logic. Note that .json files need to be included in the export options while exporting the extension from Godot.
Enabling an extension means that the scenes found in the extension.json's "nodes" array get instantiated, and disabling gets rid of these nodes from Pixelorama's SceneTree.
2022-02-19 01:21:08 +00:00
|
|
|
|
|
|
|
if save_to_config:
|
|
|
|
Global.config_cache.set_value("extensions", extension.file_name, extension.enabled)
|
|
|
|
Global.config_cache.save("user://cache.ini")
|
|
|
|
|
|
|
|
|
|
|
|
func _on_InstalledExtensions_item_selected(index: int) -> void:
|
|
|
|
extension_selected = index
|
|
|
|
var file_name: String = extension_list.get_item_metadata(extension_selected)
|
|
|
|
var extension: Extension = extensions[file_name]
|
|
|
|
if extension.enabled:
|
|
|
|
enable_button.text = "Disable"
|
|
|
|
else:
|
|
|
|
enable_button.text = "Enable"
|
|
|
|
enable_button.disabled = false
|
|
|
|
uninstall_button.disabled = false
|
|
|
|
|
|
|
|
|
|
|
|
func _on_InstalledExtensions_nothing_selected() -> void:
|
|
|
|
enable_button.disabled = true
|
|
|
|
uninstall_button.disabled = true
|
|
|
|
|
|
|
|
|
|
|
|
func _on_AddExtensionButton_pressed() -> void:
|
|
|
|
Global.preferences_dialog.get_node("Popups/AddExtensionFileDialog").popup_centered()
|
|
|
|
|
|
|
|
|
|
|
|
func _on_EnableButton_pressed() -> void:
|
|
|
|
var file_name: String = extension_list.get_item_metadata(extension_selected)
|
|
|
|
var extension: Extension = extensions[file_name]
|
|
|
|
extension.enabled = !extension.enabled
|
|
|
|
_enable_extension(extension)
|
|
|
|
if extension.enabled:
|
|
|
|
enable_button.text = "Disable"
|
|
|
|
else:
|
|
|
|
enable_button.text = "Enable"
|
|
|
|
|
|
|
|
|
|
|
|
func _on_UninstallButton_pressed() -> void:
|
2022-07-05 21:12:38 +00:00
|
|
|
_uninstall_extension(extension_list.get_item_metadata(extension_selected))
|
Implement a basic extension system
Importing .pck or .zip Godot resource pack files into Pixelorama is now possible. This needs to be documented properly, but here's the basic idea, for now at least. This is super early work and I haven't tested it with a proper extension yet, so all of this could be a subject of change. I tested it with a custom theme extension though and it seems to be working perfectly.
Importing resource pack files, either by dragging and dropping them into the app window or by going to Edit>Preferences>Extensions>Add Extension, copies the files into user://extensions/. Extensions can be enabled/disabled and uninstalled. Uninstalling them deletes the resource pack files from user://extensions/.
The extension project source files need to be in a folder inside src/Extensions/ with the same name as the .pck or .zip file. **This is required for now, otherwise it will not work.** Inside that folder there also needs to be an extension.json file, with a structure similar to this:
{
"name": "ExtensionName",
"display_name": "Extension Name",
"description": "A Pixelorama extension",
"author": "Orama Interactive",
"version": "0.1",
"license": "MIT",
"nodes": [
"ExtensionExample.tscn"
]
}
The `nodes` array leads to the packed scene files with the nodes that are to be instantiated. **The root nodes of these scenes need to have the same name as the .tscn files they belong to.** The scripts of these nodes should have _enter_tree() and _exit_tree() methods to handle the extension enabling/disabling (or even uninstalling) logic. Note that .json files need to be included in the export options while exporting the extension from Godot.
Enabling an extension means that the scenes found in the extension.json's "nodes" array get instantiated, and disabling gets rid of these nodes from Pixelorama's SceneTree.
2022-02-19 01:21:08 +00:00
|
|
|
|
|
|
|
|
|
|
|
func _on_OpenFolderButton_pressed() -> void:
|
|
|
|
OS.shell_open(ProjectSettings.globalize_path(EXTENSIONS_PATH))
|
|
|
|
|
|
|
|
|
|
|
|
func _on_AddExtensionFileDialog_files_selected(paths: PoolStringArray) -> void:
|
|
|
|
for path in paths:
|
|
|
|
install_extension(path)
|