Merge pull request #137 from Yusarina/Current-Dev
Auto load now works with custom blender extensions folders
This commit is contained in:
+5
-20
@@ -13,38 +13,23 @@ def show_version_error_popup():
|
|||||||
bpy.context.window_manager.popup_menu(draw, title="Avatar Toolkit Version Error", icon='ERROR')
|
bpy.context.window_manager.popup_menu(draw, title="Avatar Toolkit Version Error", icon='ERROR')
|
||||||
|
|
||||||
def register():
|
def register():
|
||||||
# Check Blender version first
|
|
||||||
import bpy
|
import bpy
|
||||||
version = bpy.app.version
|
version = bpy.app.version
|
||||||
if version[0] > 4 or (version[0] == 4 and version[1] >= 5):
|
if version[0] > 4 or (version[0] == 4 and version[1] >= 5):
|
||||||
show_version_error_popup()
|
show_version_error_popup()
|
||||||
return
|
return
|
||||||
|
|
||||||
# Add wheel installation check
|
|
||||||
try:
|
|
||||||
import lz4
|
|
||||||
except ImportError:
|
|
||||||
import sys
|
|
||||||
import os
|
|
||||||
import site
|
|
||||||
import pip
|
|
||||||
wheels_dir = os.path.join(os.path.dirname(__file__), "wheels")
|
|
||||||
for wheel in os.listdir(wheels_dir):
|
|
||||||
if wheel.endswith(".whl"):
|
|
||||||
pip.main(['install', os.path.join(wheels_dir, wheel)])
|
|
||||||
site.addsitedir(site.getsitepackages()[0])
|
|
||||||
|
|
||||||
from .core import auto_load
|
|
||||||
print("Starting registration")
|
print("Starting registration")
|
||||||
|
|
||||||
# Make sure to initialize logging first
|
# Import modules using relative imports
|
||||||
|
from . import core
|
||||||
|
from .core import auto_load
|
||||||
from .core.logging_setup import configure_logging
|
from .core.logging_setup import configure_logging
|
||||||
|
|
||||||
|
# Initialize logging
|
||||||
configure_logging(False)
|
configure_logging(False)
|
||||||
|
|
||||||
# Then initialize the addon
|
|
||||||
auto_load.init()
|
auto_load.init()
|
||||||
|
|
||||||
# Register classes in proper order
|
|
||||||
auto_load.register()
|
auto_load.register()
|
||||||
|
|
||||||
# Verify property registration
|
# Verify property registration
|
||||||
|
|||||||
+59
-62
@@ -1,6 +1,5 @@
|
|||||||
import os
|
import os
|
||||||
import bpy
|
import bpy
|
||||||
import sys
|
|
||||||
import typing
|
import typing
|
||||||
import inspect
|
import inspect
|
||||||
import pkgutil
|
import pkgutil
|
||||||
@@ -24,7 +23,6 @@ def init() -> None:
|
|||||||
global modules
|
global modules
|
||||||
global ordered_classes
|
global ordered_classes
|
||||||
|
|
||||||
# Configure logging first
|
|
||||||
from .logging_setup import configure_logging
|
from .logging_setup import configure_logging
|
||||||
configure_logging(False)
|
configure_logging(False)
|
||||||
|
|
||||||
@@ -32,14 +30,24 @@ def init() -> None:
|
|||||||
configure_logging(get_preference("enable_logging", False))
|
configure_logging(get_preference("enable_logging", False))
|
||||||
|
|
||||||
print("Auto-load init starting")
|
print("Auto-load init starting")
|
||||||
modules = get_all_submodules(Path(__file__).parent.parent)
|
|
||||||
|
package_name = __package__.rsplit('.', 1)[0]
|
||||||
|
directory = Path(__file__).parent.parent
|
||||||
|
modules = get_all_submodules(directory, package_name)
|
||||||
ordered_classes = get_ordered_classes_to_register(modules)
|
ordered_classes = get_ordered_classes_to_register(modules)
|
||||||
print(f"Found modules: {modules}")
|
print(f"Found modules: {modules}")
|
||||||
print(f"Found classes: {ordered_classes}")
|
print(f"Found classes: {ordered_classes}")
|
||||||
|
|
||||||
def register() -> None:
|
def register() -> None:
|
||||||
"""Register all discovered classes and modules"""
|
"""Register all discovered classes and modules"""
|
||||||
|
global modules, ordered_classes
|
||||||
|
|
||||||
print("Registering classes")
|
print("Registering classes")
|
||||||
|
|
||||||
|
if not ordered_classes:
|
||||||
|
print("Warning: No classes to register")
|
||||||
|
ordered_classes = []
|
||||||
|
|
||||||
for cls in ordered_classes:
|
for cls in ordered_classes:
|
||||||
print(f"Registering: {cls}")
|
print(f"Registering: {cls}")
|
||||||
try:
|
try:
|
||||||
@@ -47,6 +55,10 @@ def register() -> None:
|
|||||||
except ValueError:
|
except ValueError:
|
||||||
continue
|
continue
|
||||||
|
|
||||||
|
if not modules:
|
||||||
|
print("Warning: No modules to register")
|
||||||
|
modules = []
|
||||||
|
|
||||||
for module in modules:
|
for module in modules:
|
||||||
if module.__name__ == __name__:
|
if module.__name__ == __name__:
|
||||||
continue
|
continue
|
||||||
@@ -67,44 +79,29 @@ def unregister() -> None:
|
|||||||
if hasattr(module, "unregister"):
|
if hasattr(module, "unregister"):
|
||||||
module.unregister()
|
module.unregister()
|
||||||
|
|
||||||
def get_manifest_id() -> str:
|
def get_all_submodules(directory: Path, package_name: str) -> List[Any]:
|
||||||
"""Get the addon ID from the manifest file"""
|
|
||||||
manifest_path = Path(__file__).parent.parent / "blender_manifest.toml"
|
|
||||||
with open(manifest_path, "rb") as f:
|
|
||||||
manifest = tomllib.load(f)
|
|
||||||
return manifest["id"]
|
|
||||||
|
|
||||||
def get_all_submodules(directory: Path) -> List[Any]:
|
|
||||||
"""Discover and import all submodules in the given directory"""
|
"""Discover and import all submodules in the given directory"""
|
||||||
modules = []
|
return list(iter_submodules(directory, package_name))
|
||||||
addon_id = get_manifest_id()
|
|
||||||
for root, dirs, files in os.walk(directory):
|
|
||||||
if "__pycache__" in root:
|
|
||||||
continue
|
|
||||||
path = Path(root)
|
|
||||||
if path == directory:
|
|
||||||
package_name = f"bl_ext.user_default.{addon_id}"
|
|
||||||
else:
|
|
||||||
relative_path = path.relative_to(directory).as_posix().replace('/', '.')
|
|
||||||
package_name = f"bl_ext.user_default.{addon_id}.{relative_path}"
|
|
||||||
for name in sorted(iter_module_names(path)):
|
|
||||||
modules.append(importlib.import_module(f".{name}", package_name))
|
|
||||||
return modules
|
|
||||||
|
|
||||||
def iter_submodules(path: Path, package_name: str) -> Generator[Any, None, None]:
|
def iter_submodules(directory: Path, package_name: str) -> Generator[Any, None, None]:
|
||||||
"""Iterate through submodules in a package"""
|
"""Iterate through submodules in a package"""
|
||||||
for name in sorted(iter_module_names(path)):
|
for name in sorted(iter_submodule_names(directory)):
|
||||||
|
try:
|
||||||
yield importlib.import_module("." + name, package_name)
|
yield importlib.import_module("." + name, package_name)
|
||||||
|
print(f"Successfully imported {name} from {package_name}")
|
||||||
|
except ImportError as e:
|
||||||
|
print(f"Error importing {name} from {package_name}: {e}")
|
||||||
|
|
||||||
def iter_module_names(path: Path) -> Generator[str, None, None]:
|
def iter_submodule_names(path: Path, root: str = "") -> Generator[str, None, None]:
|
||||||
"""Iterate through module names in a directory"""
|
"""Iterate through module names in a directory"""
|
||||||
print(f"Scanning path: {path}")
|
print(f"Scanning path: {path}")
|
||||||
modules_list = list(pkgutil.iter_modules([str(path)]))
|
for _, module_name, is_package in pkgutil.iter_modules([str(path)]):
|
||||||
print(f"Found these modules: {modules_list}")
|
if is_package:
|
||||||
for _, module_name, is_pkg in modules_list:
|
sub_path = path / module_name
|
||||||
if not is_pkg:
|
sub_root = root + module_name + "."
|
||||||
print(f"Found module: {module_name}")
|
yield from iter_submodule_names(sub_path, sub_root)
|
||||||
yield module_name
|
else:
|
||||||
|
yield root + module_name
|
||||||
|
|
||||||
def get_ordered_classes_to_register(modules: List[Any]) -> List[Type]:
|
def get_ordered_classes_to_register(modules: List[Any]) -> List[Type]:
|
||||||
"""Get a topologically sorted list of classes to register"""
|
"""Get a topologically sorted list of classes to register"""
|
||||||
@@ -112,28 +109,37 @@ def get_ordered_classes_to_register(modules: List[Any]) -> List[Type]:
|
|||||||
|
|
||||||
def get_register_deps_dict(modules: List[Any]) -> Dict[Type, Set[Type]]:
|
def get_register_deps_dict(modules: List[Any]) -> Dict[Type, Set[Type]]:
|
||||||
"""Get dependencies dictionary for class registration"""
|
"""Get dependencies dictionary for class registration"""
|
||||||
|
my_classes = set(iter_classes_to_register(modules))
|
||||||
|
my_classes_by_idname = {cls.bl_idname: cls for cls in my_classes if hasattr(cls, "bl_idname")}
|
||||||
|
|
||||||
deps_dict = {}
|
deps_dict = {}
|
||||||
classes_to_register = set(iter_classes_to_register(modules))
|
for cls in my_classes:
|
||||||
for cls in classes_to_register:
|
deps_dict[cls] = set()
|
||||||
deps_dict[cls] = set(iter_own_register_deps(cls, classes_to_register))
|
deps_dict[cls].update(iter_deps_from_annotations(cls, my_classes))
|
||||||
|
deps_dict[cls].update(iter_deps_from_parent_id(cls, my_classes_by_idname))
|
||||||
|
|
||||||
return deps_dict
|
return deps_dict
|
||||||
|
|
||||||
def iter_own_register_deps(cls: Type, classes_to_register: Set[Type]) -> Generator[Type, None, None]:
|
def iter_deps_from_annotations(cls: Type, my_classes: Set[Type]) -> Generator[Type, None, None]:
|
||||||
"""Iterate through a class's own registration dependencies"""
|
"""Iterate through dependencies from class annotations"""
|
||||||
yield from (dep for dep in iter_register_deps(cls) if dep in classes_to_register)
|
|
||||||
|
|
||||||
def iter_register_deps(cls: Type) -> Generator[Type, None, None]:
|
|
||||||
"""Iterate through all registration dependencies of a class"""
|
|
||||||
for value in typing.get_type_hints(cls, {}, {}).values():
|
for value in typing.get_type_hints(cls, {}, {}).values():
|
||||||
dependency = get_dependency_from_annotation(value)
|
dependency = get_dependency_from_annotation(value)
|
||||||
if dependency is not None:
|
if dependency is not None and dependency in my_classes:
|
||||||
yield dependency
|
yield dependency
|
||||||
|
|
||||||
|
def iter_deps_from_parent_id(cls: Type, my_classes_by_idname: Dict[str, Type]) -> Generator[Type, None, None]:
|
||||||
|
"""Iterate through dependencies from panel parent IDs"""
|
||||||
|
if bpy.types.Panel in cls.__bases__:
|
||||||
|
parent_idname = getattr(cls, "bl_parent_id", None)
|
||||||
|
if parent_idname is not None:
|
||||||
|
parent_cls = my_classes_by_idname.get(parent_idname)
|
||||||
|
if parent_cls is not None:
|
||||||
|
yield parent_cls
|
||||||
|
|
||||||
def get_dependency_from_annotation(value: Any) -> Optional[Type]:
|
def get_dependency_from_annotation(value: Any) -> Optional[Type]:
|
||||||
"""Get dependency type from a type annotation"""
|
"""Get dependency type from a type annotation"""
|
||||||
if isinstance(value, tuple) and len(value) == 2:
|
if isinstance(value, bpy.props._PropertyDeferred):
|
||||||
if value[0] in (bpy.props.PointerProperty, bpy.props.CollectionProperty):
|
return value.keywords.get("type")
|
||||||
return value[1]["type"]
|
|
||||||
return None
|
return None
|
||||||
|
|
||||||
def iter_classes_to_register(modules: List[Any]) -> Generator[Type, None, None]:
|
def iter_classes_to_register(modules: List[Any]) -> Generator[Type, None, None]:
|
||||||
@@ -164,7 +170,8 @@ def get_register_base_types() -> Set[Type]:
|
|||||||
"Panel", "Operator", "PropertyGroup",
|
"Panel", "Operator", "PropertyGroup",
|
||||||
"AddonPreferences", "Header", "Menu",
|
"AddonPreferences", "Header", "Menu",
|
||||||
"Node", "NodeSocket", "NodeTree",
|
"Node", "NodeSocket", "NodeTree",
|
||||||
"UIList", "RenderEngine"
|
"UIList", "RenderEngine",
|
||||||
|
"Gizmo", "GizmoGroup",
|
||||||
])
|
])
|
||||||
|
|
||||||
def toposort(deps_dict: Dict[Type, Set[Type]]) -> List[Type]:
|
def toposort(deps_dict: Dict[Type, Set[Type]]) -> List[Type]:
|
||||||
@@ -172,25 +179,15 @@ def toposort(deps_dict: Dict[Type, Set[Type]]) -> List[Type]:
|
|||||||
sorted_list = []
|
sorted_list = []
|
||||||
sorted_values = set()
|
sorted_values = set()
|
||||||
|
|
||||||
panels_to_sort = [(value, deps) for value, deps in deps_dict.items()
|
while len(deps_dict) > 0:
|
||||||
if hasattr(value, 'bl_parent_id')]
|
|
||||||
|
|
||||||
base_panels = [(value, deps) for value, deps in deps_dict.items()
|
|
||||||
if not hasattr(value, 'bl_parent_id')]
|
|
||||||
|
|
||||||
for value, deps in base_panels:
|
|
||||||
if len(deps) == 0:
|
|
||||||
sorted_list.append(value)
|
|
||||||
sorted_values.add(value)
|
|
||||||
|
|
||||||
while len(deps_dict) > len(sorted_values):
|
|
||||||
unsorted = []
|
unsorted = []
|
||||||
for value, deps in deps_dict.items():
|
for value, deps in deps_dict.items():
|
||||||
if value not in sorted_values:
|
if len(deps) == 0:
|
||||||
if len(deps - sorted_values) == 0:
|
|
||||||
sorted_list.append(value)
|
sorted_list.append(value)
|
||||||
sorted_values.add(value)
|
sorted_values.add(value)
|
||||||
else:
|
else:
|
||||||
unsorted.append(value)
|
unsorted.append(value)
|
||||||
|
|
||||||
|
deps_dict = {value: deps_dict[value] - sorted_values for value in unsorted}
|
||||||
|
|
||||||
return sorted_list
|
return sorted_list
|
||||||
|
|||||||
Reference in New Issue
Block a user