From 5f3021501d15732ac663070f339a4724304a6fc3 Mon Sep 17 00:00:00 2001 From: SimonNordon4 Date: Mon, 5 Aug 2024 20:18:08 +1000 Subject: [PATCH] testing relative imports in plugin --- .gitignore | 2 + entry_point.py | 11 +++ hello-world.zip | Bin 1596 -> 1805 bytes hello-world/__init__.py | 3 + hello-world/blender_manifest.toml | 1 + hello-world/util.py | 2 + index.html | 4 +- index.json | 4 +- readme.md | 17 +++- template/__init__.py | 14 +++ template/auto_load.py | 152 ++++++++++++++++++++++++++++++ template/blender_manifest.toml | 74 +++++++++++++++ template/template_dev_panel.py | 33 +++++++ 13 files changed, 312 insertions(+), 5 deletions(-) create mode 100644 .gitignore create mode 100644 entry_point.py create mode 100644 hello-world/util.py create mode 100644 template/__init__.py create mode 100644 template/auto_load.py create mode 100644 template/blender_manifest.toml create mode 100644 template/template_dev_panel.py diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..9115925 --- /dev/null +++ b/.gitignore @@ -0,0 +1,2 @@ + +*.pyc diff --git a/entry_point.py b/entry_point.py new file mode 100644 index 0000000..6deefff --- /dev/null +++ b/entry_point.py @@ -0,0 +1,11 @@ +import importlib +import os +import sys + +script_dir = os.path.dirname(os.path.abspath(__file__)) +sys.path.insert(0, script_dir) +__package__ = os.path.basename(script_dir) + +from .template import __init__ + +__init__.test() \ No newline at end of file diff --git a/hello-world.zip b/hello-world.zip index 7ed75fdf45d988a6752663417c1adebd047f41f7..06c5d84713c5309f74188b56132fdb88a9eab475 100644 GIT binary patch delta 1086 zcmdnP)5|9v;LXe;!oa}5!Jxa4HL@zVbKg@I1_oVj1_s5+ds$UB+DkIlzdm05K2tWt z$8xLTt_^Ee?c`k;lfKRC_4ZId%aAG!jp7)K$or>sL*uV8UXs4Gptd)H-}vPv6Qv34 z;V0TMPWlChq}1BWUAcUEol>yr7UwT>O-xPQZ-y=~nv#1ZHpz5BTYtBF@1>Om?|$+A zy(_%u&m-Y{GaJj0^^Ev8?N5hUSxnRj?BIVUu_AbijQu(WM zoR>bC%nb^DCgEAO)g-A`^P<+Kq#ee=hpi+TCpyWwavyrg;Jl>u)&aRgytW%_%(t0K z{_bXa+Q|K9)t))WKd)WY=)P!M+s0#@vpqQ0SuS~2ep4spkthG_qi285x6^6;?D6r# zm->lYd~Yb1K33jbUZr#_VE>-_=!U6>ekpBDF08Bi^fDsZ`=R5yPmu`+ldh!4Y+Jon zde5zGhL7iQD#XXe#E6EAho3*{zh&_`t1Uler?E7vH=h>Q-zT_!P1z+LclG&llgz)} z_Icyut8sMpS(jHo>z`~2k3Bx;wCuHZm-8%N)IatKPrl5g&c1dt`=!HvvnDe#S4eM< z;k|r-D^Yf7^O4ssCQ3GQ@9?AK4|F2Bo4x5i> zMq_(4?@opA43`Rm&e~+f@;+wUp7N>O66P)a zllUI*P;Ei}uU+@!u~X~ql znngarAS)~~zHYXanOCxn_75KIeCe*68$BYI@Gbiv^nOlHZp6if?|!v~RV1a#>Mp)| zir19Ib#C;l4y$}u2%4A|dSSACs?4DW+Z!FM`HXs&9vnxJ6bFx1xr5m)dQFaCRk!Tm@t7T# z30-!mhY@HOBa;XNw){DH4Xa}!x^F?AMF166tX@Yp6PphZ225a`Y{x1o7U0dw1`=cj L!mB`93Csfk&@aG> delta 854 zcmeC>+ruLr;LXe;!oa}5!7#gqHIkcI-Rdz51H*eR1_s5+zHBNR?IjuOv%lTDY_(!W zz-@`xh_I_M9vato-e3ClwiY|@R5`9wI--8pwikI{oj$W+Tbo&igYvb*KC?}FIi6h1 zO>y3mvRqSZ(>{6rKi@a4J`uwER-jhh*Y|8-ndk|QRO!;`6`U&`K7QoBxI}Bt?=t85 z?=1TJe;kqQdolT2Sd{9q`P0KYKN{B;h94?AS2@WoC)G!4wy?wEg{B+ouP05~7hLWq z(7ycTlvzf{Jr1p0DwzEuX7!0V#p%o1N{yJ5T5Y<-S`ITFbKu_EXuCwbDv2>KC*SCs zIO{29@gJd;mi#xOLz(+sa(R<`jg2M>#1wm+eZMJk#gWN!*Scqax37)hdOq>d!^|SuTrtsL~b56^OHMpE-`J(=@Pq;B{@^vOvcHJfHmkx)`oXpHzA^kms z_woU*MA@ayM_#*_DA~-t!{Zv`wEd>wbhgRUm~9vfCtqgP0Fr;0UF!G!W!=sgEEM(N zVVJYt2~nxW9n}UW7fxj@ej03eS0U}t!#`Ht>&5@o9GEg|4c|fQ>ALS@pE4ceeYkni zVT-&@MTw`6V{;XK|Ep-`{3}0lUwB2Xf$6bnfnA-K-P~rLyVN@8-TpO+KlXawb=?26 zzvAzw<{K5Owg2r1cxZO>UZ&8u_4ONaC(oJF!gX}kSK+-Tin(*y#kybp@K*SAsrrlD zL3NRyS)tda-(S|(dr9Ivm)+r0x7(7+bb`cIHox3|#acaCI;>>H?tKeI=S4)ipU&7? zVH4T5`5_+W!Od?il`OLL_1tU?QG!L1TwCFiA%MF_SpM5@%M*WJ&bgC}VPC!S&q@MdHZVZfFHCfl$%%A=bn4m1-1lqTP0Q{rP{ OU}Bg86r98gk^um9`Eo1( diff --git a/hello-world/__init__.py b/hello-world/__init__.py index 4fba2b4..163dad0 100644 --- a/hello-world/__init__.py +++ b/hello-world/__init__.py @@ -1,4 +1,7 @@ +from . import util + def register(): print("Hello World") + util.extra_print() def unregister(): print("Goodbye World") \ No newline at end of file diff --git a/hello-world/blender_manifest.toml b/hello-world/blender_manifest.toml index fb63b75..258fd50 100644 --- a/hello-world/blender_manifest.toml +++ b/hello-world/blender_manifest.toml @@ -10,6 +10,7 @@ maintainer = "Simon Nordon " # Supported types: "add-on", "theme" type = "add-on" +# Optional link to documentation, support, source files, etc # Optional link to documentation, support, source files, etc # website = "https://extensions.blender.org/add-ons/my-example-package/" diff --git a/hello-world/util.py b/hello-world/util.py new file mode 100644 index 0000000..7581498 --- /dev/null +++ b/hello-world/util.py @@ -0,0 +1,2 @@ +def extra_print(): + print("Extra print from hello-world/util.py") \ No newline at end of file diff --git a/index.html b/index.html index c12fc77..8783343 100644 --- a/index.html +++ b/index.html @@ -25,10 +25,10 @@ ~ 4.2.0 - ~ all - 1.6KB + 1.8KB -

Built 2024-08-05, 05:37

+

Built 2024-08-05, 10:17

diff --git a/index.json b/index.json index c7f1cef..d0d4e44 100644 --- a/index.json +++ b/index.json @@ -19,8 +19,8 @@ "Sequencer" ], "archive_url": "./hello-world.zip", - "archive_size": 1596, - "archive_hash": "sha256:99f4c18b16b81d392894c9bc75a020e2e90e43dda15cd3809d8a03f588148ac4" + "archive_size": 1805, + "archive_hash": "sha256:9deb8200f0a086a7029f3eb46db0ae6b40cb27347a3b37f1337ccd06d4b06e6e" } ] } \ No newline at end of file diff --git a/readme.md b/readme.md index d2b48e3..6fb763e 100644 --- a/readme.md +++ b/readme.md @@ -1 +1,16 @@ -https://docs.blender.org/manual/en/latest/advanced/extensions/creating_repository/static_repository.html +https://simonnordon4.github.io/blender-extensions/index.json + + + +blender --command extension server-generate --repo-dir=E:\repos\blender-extensions\ --html + +Compress-Archive -Path ".\hello-world" -DestinationPath "hello-world.zip" -Force + +1. Create a new directory for the extension. +2. Compres the directory into a zip file. +3. Run blender extension command. +4. Upload the zip file to the server. +5. Ensure repo is public and a website page. + +References: + https://docs.blender.org/manual/en/latest/advanced/extensions/creating_repository/static_repository.html \ No newline at end of file diff --git a/template/__init__.py b/template/__init__.py new file mode 100644 index 0000000..bf323d3 --- /dev/null +++ b/template/__init__.py @@ -0,0 +1,14 @@ +# import auto_load + +# auto_load.init() + +# def register(): +# auto_load.register() + +# def unregister(): +# auto_load.unregister() + +print("Hello from __init__.py") + +def test(): + print("Hello from __init__.py test()") \ No newline at end of file diff --git a/template/auto_load.py b/template/auto_load.py new file mode 100644 index 0000000..16da589 --- /dev/null +++ b/template/auto_load.py @@ -0,0 +1,152 @@ +import os +import bpy +import sys +import typing +import inspect +import pkgutil +import importlib +from pathlib import Path + +blender_version = bpy.app.version + +modules = None +ordered_classes = None + +def init(): + global modules, ordered_classes + modules = get_all_submodules(Path(__file__).parent) + ordered_classes = get_ordered_classes_to_register(modules) + + +def register(): + # Check if class is already registered before registering it + for cls in ordered_classes: + try: + bpy.utils.register_class(cls) + except: + print("Failed to register", cls) + + for module in modules: + if module.__name__ == __name__: + continue + if hasattr(module, "register"): + module.register() + +def unregister(): + if ordered_classes is None: + return + for cls in reversed(ordered_classes): + bpy.utils.unregister_class(cls) + + for module in modules: + if module.__name__ == __name__: + continue + if hasattr(module, "unregister"): + module.unregister() + +def get_all_submodules(directory): + return list(iter_submodules(directory, directory.name)) + +def iter_submodules(path, package_name): + for name in sorted(iter_submodule_names(path)): + module = importlib.import_module(name) + try: + importlib.reload(module) + except: + pass + yield module + +def iter_submodule_names(path, root=""): + for _, module_name, is_package in pkgutil.iter_modules([str(path)]): + if is_package: + sub_path = path / module_name + sub_root = root + module_name + "." + yield from iter_submodule_names(sub_path, sub_root) + else: + yield root + module_name + +def get_ordered_classes_to_register(modules): + return toposort(get_register_deps_dict(modules)) + +def get_register_deps_dict(modules): + my_classes = set(iter_my_classes(modules)) + my_classes_by_idname = {cls.bl_idname : cls for cls in my_classes if hasattr(cls, "bl_idname")} + + deps_dict = {} + for cls in my_classes: + deps_dict[cls] = set(iter_my_register_deps(cls, my_classes, my_classes_by_idname)) + return deps_dict + +def iter_my_register_deps(cls, my_classes, my_classes_by_idname): + yield from iter_my_deps_from_annotations(cls, my_classes) + yield from iter_my_deps_from_parent_id(cls, my_classes_by_idname) + +def iter_my_deps_from_annotations(cls, my_classes): + for value in typing.get_type_hints(cls, {}, {}).values(): + dependency = get_dependency_from_annotation(value) + if dependency is not None: + if dependency in my_classes: + yield dependency + +def get_dependency_from_annotation(value): + if blender_version >= (2, 93): + if isinstance(value, bpy.props._PropertyDeferred): + return value.keywords.get("type") + else: + if isinstance(value, tuple) and len(value) == 2: + if value[0] in (bpy.props.PointerProperty, bpy.props.CollectionProperty): + return value[1]["type"] + return None + +def iter_my_deps_from_parent_id(cls, my_classes_by_idname): + 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 iter_my_classes(modules): + base_types = get_register_base_types() + for cls in get_classes_in_modules(modules): + if any(base in base_types for base in cls.__bases__): + if not getattr(cls, "is_registered", False): + yield cls + else: + bpy.utils.unregister_class(cls) + yield cls + +def get_classes_in_modules(modules): + classes = set() + for module in modules: + for cls in iter_classes_in_module(module): + classes.add(cls) + return classes + +def iter_classes_in_module(module): + for value in module.__dict__.values(): + if inspect.isclass(value): + yield value + +def get_register_base_types(): + return set(getattr(bpy.types, name) for name in [ + "Panel", "Operator", "PropertyGroup", + "AddonPreferences", "Header", "Menu", + "Node", "NodeSocket", "NodeTree", + "UIList", "RenderEngine", + "Gizmo", "GizmoGroup", + ]) + +def toposort(deps_dict): + sorted_list = [] + sorted_values = set() + while len(deps_dict) > 0: + unsorted = [] + for value, deps in deps_dict.items(): + if len(deps) == 0: + sorted_list.append(value) + sorted_values.add(value) + else: + unsorted.append(value) + deps_dict = {value : deps_dict[value] - sorted_values for value in unsorted} + return sorted_list \ No newline at end of file diff --git a/template/blender_manifest.toml b/template/blender_manifest.toml new file mode 100644 index 0000000..70e2f54 --- /dev/null +++ b/template/blender_manifest.toml @@ -0,0 +1,74 @@ +schema_version = "1.0.0" + +# Example of manifest file for a Blender extension +# Change the values according to your extension +id = "template" +version = "1.0.0" +name = "Template" +tagline = "Template for a Blender Addon" +maintainer = "Simon Nordon " +# Supported types: "add-on", "theme" +type = "add-on" + +# Optional link to documentation, support, source files, etc +# Optional link to documentation, support, source files, etc +# website = "https://extensions.blender.org/add-ons/my-example-package/" + +# Optional list defined by Blender and server, see: +# https://docs.blender.org/manual/en/dev/advanced/extensions/tags.html +tags = ["Object"] + +blender_version_min = "4.2.0" +# # Optional: Blender version that the extension does not support, earlier versions are supported. +# # This can be omitted and defined later on the extensions platform if an issue is found. +# blender_version_max = "5.1.0" + +# License conforming to https://spdx.org/licenses/ (use "SPDX: prefix) +# https://docs.blender.org/manual/en/dev/advanced/extensions/licenses.html +license = [ + "SPDX:GPL-2.0-or-later", +] +# Optional: required by some licenses. +# copyright = [ +# "2002-2024 Developer Name", +# "1998 Company Name", +# ] + +# Optional list of supported platforms. If omitted, the extension will be available in all operating systems. +# platforms = ["windows-x64", "macos-arm64", "linux-x64"] +# Other supported platforms: "windows-arm64", "macos-x64" + +# Optional: bundle 3rd party Python modules. +# https://docs.blender.org/manual/en/dev/advanced/extensions/python_wheels.html +# wheels = [ +# "./wheels/hexdump-3.3-py3-none-any.whl", +# "./wheels/jsmin-3.0.1-py3-none-any.whl", +# ] + +# # Optional: add-ons can list which resources they will require: +# # * files (for access of any filesystem operations) +# # * network (for internet access) +# # * clipboard (to read and/or write the system clipboard) +# # * camera (to capture photos and videos) +# # * microphone (to capture audio) +# # +# # If using network, remember to also check `bpy.app.online_access` +# # https://docs.blender.org/manual/en/dev/advanced/extensions/addons.html#internet-access +# # +# # For each permission it is important to also specify the reason why it is required. +# # Keep this a single short sentence without a period (.) at the end. +# # For longer explanations use the documentation or detail page. +# +# [permissions] +# network = "Need to sync motion-capture data to server" +# files = "Import/export FBX from/to disk" +# clipboard = "Copy and paste bone transforms" + +# Optional: build settings. +# https://docs.blender.org/manual/en/dev/advanced/extensions/command_line_arguments.html#command-line-args-extension-build +# [build] +# paths_exclude_pattern = [ +# "__pycache__/", +# "/.git/", +# "/*.zip", +# ] \ No newline at end of file diff --git a/template/template_dev_panel.py b/template/template_dev_panel.py new file mode 100644 index 0000000..bad361f --- /dev/null +++ b/template/template_dev_panel.py @@ -0,0 +1,33 @@ +import bpy +import auto_load + +class TemplateDevPanel(bpy.types.Panel): + bl_idname = "DEV_PT_template_dev_panel" + bl_label = "Template Dev Panel" + bl_space_type = "VIEW_3D" + bl_region_type = "UI" + bl_category = "Template" + + def draw(self, context): + layout = self.layout + layout.label(text="Template Panel", icon="MODIFIER") + layout.operator("dev_panel.reload_addon") + layout.operator("dev_panel.uninstall_addon") + +class ReloadAddonOperator(bpy.types.Operator): + bl_idname = "dev_panel.reload_addon" + bl_label = "Reload Addon" + + def execute(self, context): + auto_load.unregister() + auto_load.init() + auto_load.register() + return {"FINISHED"} + +class UninstallAddonOperator(bpy.types.Operator): + bl_idname = "dev_panel.uninstall_addon" + bl_label = "Uninstall Addon" + + def execute(self, context): + auto_load.unregister() + return {"FINISHED"}