Module sverchok.utils.sv_default_macros
Expand source code
# ##### BEGIN GPL LICENSE BLOCK #####
#
#  This program is free software; you can redistribute it and/or
#  modify it under the terms of the GNU General Public License
#  as published by the Free Software Foundation; either version 2
#  of the License, or (at your option) any later version.
#
#  This program is distributed in the hope that it will be useful,
#  but WITHOUT ANY WARRANTY; without even the implied warranty of
#  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
#  GNU General Public License for more details.
#
#  You should have received a copy of the GNU General Public License
#  along with this program; if not, write to the Free Software Foundation,
#  Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
#
# ##### END GPL LICENSE BLOCK #####
import webbrowser
import os
import urllib
import urllib.request
from zipfile import ZipFile
import tempfile
import bpy
from sverchok.utils.sv_update_utils import sv_get_local_path
from sverchok.utils.macros.math_macros import math_macros
from sverchok.utils.macros.join_macros import join_macros
from sverchok.utils.macros.switch_macros import switch_macros
from sverchok.utils.macros.gp_macros import gp_macro_one, gp_macro_two
from sverchok.utils.macros.hotswap_macros import swap_vd_mv
from sverchok.utils.macros.get_objects_data import objdata_macro_one
# pylint: disable=c0301
def simple_macro(description="", term="", macro_handler="verbose_macro_handler"):
    return {
        'display_name': description,
        'file': 'macro',
        'ident': [macro_handler, term]}    
macros = {
    "> obj vd": simple_macro(description="active_obj into objlite + vdmk2", term='obj vd'),
    "> objs vd": simple_macro(description="multi obj in + vdmk2", term='objs vd'),
    "> objs socket to get-Obj-Data": simple_macro(description="objsocket -> get objects data", term='objs socket to data'),
    "> zen": simple_macro(description="zen of Sverchok", term='zen'),
    "> nuke python++": simple_macro(description="like f8", term='nuke python++'),
    "> 3d cursor to Vector In": simple_macro(description="makes new node", term="3dcursor_to_vector_in"),
    "> 3d cursor to Matrix Input": simple_macro(description="makes new node", term="3dcursor_to_matrix"),
    "> sn petal": simple_macro(description="load snlite w/ petalsine", term='snl demo/petal_sine.py'),
    "> Subdiv to quads": simple_macro(description="snlite w/ subdiv to quads", term='snl demo/subidivide_to_quads.py'),
    "> multiply *": simple_macro(description="multiply selected nodes", term='mathMUL'),
    "> add +": simple_macro(description="add selected nodes", term='mathADD'),
    "> sub -": simple_macro(description="subtract selected nodes", term='mathSUB'),
    "> join1": simple_macro(description="selected nodes to List Join", term='join1'),
    "> join123": simple_macro(description="selected nodes to List Join", term='join123'),
    "> join12": simple_macro(description="selected nodes to List Join", term='join12'),
    "> join13": simple_macro(description="selected nodes to List Join", term='join13'),
    "> rnd col": simple_macro(description="list of n-random colors", term='rndcol'),
    "> sw1": simple_macro(description="connect nodes to switch", term='switch1'),
    "> sw12": simple_macro(description="connect nodes to switch", term='switch12'),
    "> sw13": simple_macro(description="connect nodes to switch", term='switch13'),
    "> sw123": simple_macro(description="connect nodes to switch", term='switch123'),
    "> gp +": simple_macro(description="grease pencil setup", term='gp +'),
    "> gp + 2": simple_macro(description="grease pencil setup", term='gp + 2'),
    "> hotswap vd mv": simple_macro(description="hotswap vd->meshviewer", term='hotswap'),    
    "> url": simple_macro(description="download archive from url", term='url'),
    "> blend 2 zip": simple_macro(description="archive blend as zip", term='blend 2 zip'),
    "> all numpy True": simple_macro(description="existing nodes to numpy", term='output numpy True'),
    "> all numpy False": simple_macro(description="existing nodes to python", term='output numpy False')
}
def sn_loader(snlite, script_name=None):
    sv_path = os.path.dirname(sv_get_local_path()[0])
    snlite_template_path = os.path.join(sv_path, 'node_scripts', 'SNLite_templates')
    fullpath = os.path.join(snlite_template_path, script_name)
    txt = bpy.data.texts.load(fullpath)
    snlite.script_name = os.path.basename(txt.name)
    snlite.load()
def set_node_props(node, prop_dict):
    for prop, val in prop_dict.items():
        setattr(node, prop, val)
class DefaultMacros():
    @classmethod
    def ensure_nodetree(cls, operator, context):
        '''
        if no active nodetree
        add new empty node tree, set fakeuser immediately
        '''
        if not context.space_data.tree_type in {'SverchCustomTreeType', }:
            print('not running from a sv nodetree')
            return
        if not hasattr(context.space_data.edit_tree, 'nodes'):
            msg_one = 'going to add a new empty node tree'
            msg_two = 'added new node tree'
            print(msg_one)
            operator.report({"WARNING"}, msg_one)
            ng_params = {'name': 'NodeTree', 'type': 'SverchCustomTreeType'}
            ng = bpy.data.node_groups.new(**ng_params)
            ng.use_fake_user = True
            context.space_data.node_tree = ng
            operator.report({"WARNING"}, msg_two)
    @classmethod
    def verbose_macro_handler(cls, operator, context, term):
        cls.ensure_nodetree(operator, context)
        tree = context.space_data.edit_tree
        nodes, links = tree.nodes, tree.links
        if term == 'obj vd':
            obj_in_node = nodes.new('SvObjInLite')
            obj_in_node.dget()
            vd_node = nodes.new('SvViewerDrawMk4')
            vd_node.location = obj_in_node.location.x + 180, obj_in_node.location.y
            links.new(obj_in_node.outputs[0], vd_node.inputs[0])
            links.new(obj_in_node.outputs[2], vd_node.inputs[2])
            links.new(obj_in_node.outputs[4], vd_node.inputs[3])
        elif term == 'objs vd':
            obj_in_node = nodes.new('SvGetObjectsData')
            obj_in_node.get_objects_from_scene(operator)
            vd_node = nodes.new('SvViewerDrawMk4')
            vd_node.location = obj_in_node.location.x + 180, obj_in_node.location.y
            # this macro could detect specifically if the node found edges or faces or both... 
            links.new(obj_in_node.outputs[0], vd_node.inputs[0])
            links.new(obj_in_node.outputs[2], vd_node.inputs[2])
            links.new(obj_in_node.outputs[8], vd_node.inputs[3])
        elif term == "objs socket to data":
            objdata_macro_one(context, operator, term, nodes, links)
        elif term == "3dcursor_to_vector_in":
            
            MOUSE_X, MOUSE_Y = context.space_data.cursor_location
            cursor = context.scene.cursor.location
            node = nodes.new("GenVectorsNode")
            node.location = MOUSE_X, MOUSE_Y
            node.x_, node.y_, node.z_ = tuple(cursor)
        elif term == "3dcursor_to_matrix":
            def flattened(matrix):
                return list(matrix[0]) + list(matrix[1]) + list(matrix[2]) + list(matrix[3])
            
            MOUSE_X, MOUSE_Y = context.space_data.cursor_location
            matrix = context.scene.cursor.matrix
            node = nodes.new("SvMatrixValueIn") # "SvMatrixInNodeMK4")
            node.location = MOUSE_X, MOUSE_Y
            node.matrix = flattened(matrix.transposed())
        elif term == 'rndcol':
            # all locations will be relative to cursor
            MOUSE_X, MOUSE_Y = context.space_data.cursor_location
            # add nodes to layout
            NUM = nodes.new("SvNumberNode")
            RR = nodes.new('NodeReroute')
            RND_0 = nodes.new('SvRndNumGen')
            RND_1 = nodes.new('SvRndNumGen')
            RND_2 = nodes.new('SvRndNumGen')
            COL = nodes.new('SvColorsInNodeMK1')
            # set locations
            COL.location = MOUSE_X + 140, MOUSE_Y + 40
            RND_0.location.x = MOUSE_X - 40
            RND_0.location.y = MOUSE_Y + 40
            RND_1.location.x = MOUSE_X - 40
            RND_1.location.y = MOUSE_Y
            RND_2.location.x = MOUSE_X - 40
            RND_2.location.y = MOUSE_Y - 40
            RR.location = MOUSE_X - 110, MOUSE_Y
            NUM.location = MOUSE_X - 280, MOUSE_Y + 20
            # configure nodes
            set_node_props(NUM, {
                'selected_mode': 'int', 'int_': 30})
            set_node_props(RND_0, {
                'type_selected_mode': 'Float',
                'seed': 20, 'hide': True})
            set_node_props(RND_1, {
                'type_selected_mode': 'Float',
                'seed': 30, 'hide': True})
            set_node_props(RND_2, {
                'type_selected_mode': 'Float',
                'seed': 40, 'hide': True})
            # link nodes
            links.new(NUM.outputs[0], RR.inputs[0])
            links.new(RR.outputs[0], RND_0.inputs[0])
            links.new(RR.outputs[0], RND_1.inputs[0])
            links.new(RR.outputs[0], RND_2.inputs[0])
            links.new(RND_0.outputs[0], COL.inputs[0])
            links.new(RND_1.outputs[0], COL.inputs[1])
            links.new(RND_2.outputs[0], COL.inputs[2])
        elif 'hotswap' in term:
            swap_vd_mv(context, operator, term, nodes, links)
            return
        elif term == 'nuke python++':
            bpy.ops.script.reload()
        elif term == 'zen':
            full_url_term = 'https://gist.github.com/zeffii/d843b985b0db97af56dfa9c30cd54712'
            webbrowser.open(full_url_term)
        elif 'snl' in term:
            file = term.split(' ')[1]
            snlite = nodes.new('SvScriptNodeLite')
            snlite.location = context.space_data.cursor_location
            sn_loader(snlite, script_name=file)
        elif "join" in term:
            join_macros(context, operator, term, nodes, links)
        elif "math" in term:
            math_macros(context, operator, term, nodes, links)
            return
        elif "switch" in term:
            switch_macros(context, operator, term, nodes, links)
            return
        elif 'output numpy' in term:
            state_ = term.split(' ')[2]
            state = state_ == 'True'
            for node in nodes:
                # most of cases
                if hasattr(node, 'output_numpy'):
                    node.output_numpy = state
                # line node
                if hasattr(node, 'as_numpy'):
                    node.as_numpy = state
                # box node
                if hasattr(node, 'out_np'):
                    for i in range(len(node.out_np)):
                        node.out_np[i] = state
                # vector in
                if hasattr(node, 'implementation'):
                    try:
                        node.implementation = 'NumPy' if state else 'Python'
                    except TypeError:
                        pass
        elif term == 'gp +':
            gp_macro_one(context, operator, term, nodes, links)
        elif term == 'gp + 2':
            gp_macro_two(context, operator, term, nodes, links)
        elif term == 'url':
            bpy.ops.node.sv_load_archived_blend_url()
        elif term == 'blend 2 zip':
            bpy.ops.node.blend_to_archive(archive_ext="zip")
"""
to the reader: when we run 'bpy.ops.wm.open_mainfile( )' there's a considerations
 - the current .blend can be discarded (and up to the user to already save it)
 - the end of the operator that was executed to call 'bpy.ops.wm.open_main_file( )'
   is never reached, and it is irrelevant whether we return {`FINISHED`}.
"""
def handle_zip(self, wm, to_path, file):
    try:
        err = 0
        with ZipFile(to_path) as zf:
            inner_files = zf.namelist()
            if len(inner_files) == 1:
                blendfile = inner_files[0]
            else:
                print('cancelled, not a github zipped .blend')
                return {'CANCELLED'}
        ZipFile(file[0]).extractall(path=self.os_temp_path, members=None, pwd=None)
        wm.progress_update(90)
        err = 1
        os.remove(file[0])
        err = 2
        wm.progress_update(100)
        wm.progress_end()
        fp = os.path.join(self.os_temp_path, blendfile)
        bpy.ops.wm.open_mainfile(filepath=fp)
        return {'FINISHED'}  # it never reaches here..
    except:
        self.report({'ERROR'}, "Cannot extract files errno {0}".format(str(err)))
        wm.progress_end()
        os.remove(file[0])
        return {'CANCELLED'}
def handle_gz(self, wm, to_path):
    wm.progress_update(100)
    wm.progress_end()
    bpy.ops.wm.open_mainfile(filepath=to_path)
    return {'FINISHED'}
class SvLoadArchivedBlendURL(bpy.types.Operator):
    bl_idname = "node.sv_load_archived_blend_url"
    bl_label = "Load archive URL (.zip .gz of blend)"
    """
    loads links like:
    https://github.com/nortikin/sverchok/files/647412/blend_name_YYYY_MM_DD_HH_mm.zip
    https://github.com/nortikin/sverchok/files/647412/blend_name_YYYY_MM_DD_HH_mm.gz
    """
    download_url: bpy.props.StringProperty()
    os_temp_path = tempfile.gettempdir()
    def execute(self, context):
        wm = bpy.data.window_managers[0]
        wm.progress_begin(0, 100)
        wm.progress_update(20)
        if not self.download_url:
            clipboard = context.window_manager.clipboard
            if not clipboard:
                self.report({'ERROR'}, "Clipboard empty")
                return {'CANCELLED'}
            else:
                self.download_url = clipboard
        # get the file and obtain the new file's path on disk
        try:
            file_and_ext = os.path.basename(self.download_url)
            to_path = os.path.join(self.os_temp_path, file_and_ext)
            file = urllib.request.urlretrieve(self.download_url, to_path)
            wm.progress_update(50)
        except Exception as fullerr:
            print(repr(fullerr))
            self.report({'ERROR'}, "Cannot get archive from Internet")
            wm.progress_end()
            return {'CANCELLED'}
        if file_and_ext.endswith('.gz'):
            return handle_gz(self, wm, to_path)
        elif file_and_ext.endswith('.zip'):
            return handle_zip(self, wm, to_path, file)
        else:
            # maybe if the url ends in .blend we could load it anyway.
            self.report({'ERROR'}, "url is not a .gz or .zip... ending operator")
            wm.progress_end()
            return {'CANCELLED'}
        return {'FINISHED'}
classes = [
    SvLoadArchivedBlendURL,
]
def register():
    for cls in classes:
        bpy.utils.register_class(cls)
def unregister():
    for cls in classes[::-1]:
        bpy.utils.unregister_class(cls)
# EOFFunctions
- def handle_gz(self, wm, to_path)
- 
Expand source codedef handle_gz(self, wm, to_path): wm.progress_update(100) wm.progress_end() bpy.ops.wm.open_mainfile(filepath=to_path) return {'FINISHED'}
- def handle_zip(self, wm, to_path, file)
- 
Expand source codedef handle_zip(self, wm, to_path, file): try: err = 0 with ZipFile(to_path) as zf: inner_files = zf.namelist() if len(inner_files) == 1: blendfile = inner_files[0] else: print('cancelled, not a github zipped .blend') return {'CANCELLED'} ZipFile(file[0]).extractall(path=self.os_temp_path, members=None, pwd=None) wm.progress_update(90) err = 1 os.remove(file[0]) err = 2 wm.progress_update(100) wm.progress_end() fp = os.path.join(self.os_temp_path, blendfile) bpy.ops.wm.open_mainfile(filepath=fp) return {'FINISHED'} # it never reaches here.. except: self.report({'ERROR'}, "Cannot extract files errno {0}".format(str(err))) wm.progress_end() os.remove(file[0]) return {'CANCELLED'}
- def register()
- 
Expand source codedef register(): for cls in classes: bpy.utils.register_class(cls)
- def set_node_props(node, prop_dict)
- 
Expand source codedef set_node_props(node, prop_dict): for prop, val in prop_dict.items(): setattr(node, prop, val)
- def simple_macro(description='', term='', macro_handler='verbose_macro_handler')
- 
Expand source codedef simple_macro(description="", term="", macro_handler="verbose_macro_handler"): return { 'display_name': description, 'file': 'macro', 'ident': [macro_handler, term]}
- def sn_loader(snlite, script_name=None)
- 
Expand source codedef sn_loader(snlite, script_name=None): sv_path = os.path.dirname(sv_get_local_path()[0]) snlite_template_path = os.path.join(sv_path, 'node_scripts', 'SNLite_templates') fullpath = os.path.join(snlite_template_path, script_name) txt = bpy.data.texts.load(fullpath) snlite.script_name = os.path.basename(txt.name) snlite.load()
- def unregister()
- 
Expand source codedef unregister(): for cls in classes[::-1]: bpy.utils.unregister_class(cls)
Classes
- class DefaultMacros
- 
Expand source codeclass DefaultMacros(): @classmethod def ensure_nodetree(cls, operator, context): ''' if no active nodetree add new empty node tree, set fakeuser immediately ''' if not context.space_data.tree_type in {'SverchCustomTreeType', }: print('not running from a sv nodetree') return if not hasattr(context.space_data.edit_tree, 'nodes'): msg_one = 'going to add a new empty node tree' msg_two = 'added new node tree' print(msg_one) operator.report({"WARNING"}, msg_one) ng_params = {'name': 'NodeTree', 'type': 'SverchCustomTreeType'} ng = bpy.data.node_groups.new(**ng_params) ng.use_fake_user = True context.space_data.node_tree = ng operator.report({"WARNING"}, msg_two) @classmethod def verbose_macro_handler(cls, operator, context, term): cls.ensure_nodetree(operator, context) tree = context.space_data.edit_tree nodes, links = tree.nodes, tree.links if term == 'obj vd': obj_in_node = nodes.new('SvObjInLite') obj_in_node.dget() vd_node = nodes.new('SvViewerDrawMk4') vd_node.location = obj_in_node.location.x + 180, obj_in_node.location.y links.new(obj_in_node.outputs[0], vd_node.inputs[0]) links.new(obj_in_node.outputs[2], vd_node.inputs[2]) links.new(obj_in_node.outputs[4], vd_node.inputs[3]) elif term == 'objs vd': obj_in_node = nodes.new('SvGetObjectsData') obj_in_node.get_objects_from_scene(operator) vd_node = nodes.new('SvViewerDrawMk4') vd_node.location = obj_in_node.location.x + 180, obj_in_node.location.y # this macro could detect specifically if the node found edges or faces or both... links.new(obj_in_node.outputs[0], vd_node.inputs[0]) links.new(obj_in_node.outputs[2], vd_node.inputs[2]) links.new(obj_in_node.outputs[8], vd_node.inputs[3]) elif term == "objs socket to data": objdata_macro_one(context, operator, term, nodes, links) elif term == "3dcursor_to_vector_in": MOUSE_X, MOUSE_Y = context.space_data.cursor_location cursor = context.scene.cursor.location node = nodes.new("GenVectorsNode") node.location = MOUSE_X, MOUSE_Y node.x_, node.y_, node.z_ = tuple(cursor) elif term == "3dcursor_to_matrix": def flattened(matrix): return list(matrix[0]) + list(matrix[1]) + list(matrix[2]) + list(matrix[3]) MOUSE_X, MOUSE_Y = context.space_data.cursor_location matrix = context.scene.cursor.matrix node = nodes.new("SvMatrixValueIn") # "SvMatrixInNodeMK4") node.location = MOUSE_X, MOUSE_Y node.matrix = flattened(matrix.transposed()) elif term == 'rndcol': # all locations will be relative to cursor MOUSE_X, MOUSE_Y = context.space_data.cursor_location # add nodes to layout NUM = nodes.new("SvNumberNode") RR = nodes.new('NodeReroute') RND_0 = nodes.new('SvRndNumGen') RND_1 = nodes.new('SvRndNumGen') RND_2 = nodes.new('SvRndNumGen') COL = nodes.new('SvColorsInNodeMK1') # set locations COL.location = MOUSE_X + 140, MOUSE_Y + 40 RND_0.location.x = MOUSE_X - 40 RND_0.location.y = MOUSE_Y + 40 RND_1.location.x = MOUSE_X - 40 RND_1.location.y = MOUSE_Y RND_2.location.x = MOUSE_X - 40 RND_2.location.y = MOUSE_Y - 40 RR.location = MOUSE_X - 110, MOUSE_Y NUM.location = MOUSE_X - 280, MOUSE_Y + 20 # configure nodes set_node_props(NUM, { 'selected_mode': 'int', 'int_': 30}) set_node_props(RND_0, { 'type_selected_mode': 'Float', 'seed': 20, 'hide': True}) set_node_props(RND_1, { 'type_selected_mode': 'Float', 'seed': 30, 'hide': True}) set_node_props(RND_2, { 'type_selected_mode': 'Float', 'seed': 40, 'hide': True}) # link nodes links.new(NUM.outputs[0], RR.inputs[0]) links.new(RR.outputs[0], RND_0.inputs[0]) links.new(RR.outputs[0], RND_1.inputs[0]) links.new(RR.outputs[0], RND_2.inputs[0]) links.new(RND_0.outputs[0], COL.inputs[0]) links.new(RND_1.outputs[0], COL.inputs[1]) links.new(RND_2.outputs[0], COL.inputs[2]) elif 'hotswap' in term: swap_vd_mv(context, operator, term, nodes, links) return elif term == 'nuke python++': bpy.ops.script.reload() elif term == 'zen': full_url_term = 'https://gist.github.com/zeffii/d843b985b0db97af56dfa9c30cd54712' webbrowser.open(full_url_term) elif 'snl' in term: file = term.split(' ')[1] snlite = nodes.new('SvScriptNodeLite') snlite.location = context.space_data.cursor_location sn_loader(snlite, script_name=file) elif "join" in term: join_macros(context, operator, term, nodes, links) elif "math" in term: math_macros(context, operator, term, nodes, links) return elif "switch" in term: switch_macros(context, operator, term, nodes, links) return elif 'output numpy' in term: state_ = term.split(' ')[2] state = state_ == 'True' for node in nodes: # most of cases if hasattr(node, 'output_numpy'): node.output_numpy = state # line node if hasattr(node, 'as_numpy'): node.as_numpy = state # box node if hasattr(node, 'out_np'): for i in range(len(node.out_np)): node.out_np[i] = state # vector in if hasattr(node, 'implementation'): try: node.implementation = 'NumPy' if state else 'Python' except TypeError: pass elif term == 'gp +': gp_macro_one(context, operator, term, nodes, links) elif term == 'gp + 2': gp_macro_two(context, operator, term, nodes, links) elif term == 'url': bpy.ops.node.sv_load_archived_blend_url() elif term == 'blend 2 zip': bpy.ops.node.blend_to_archive(archive_ext="zip")Static methods- def ensure_nodetree(operator, context)
- 
if no active nodetree add new empty node tree, set fakeuser immediately Expand source code@classmethod def ensure_nodetree(cls, operator, context): ''' if no active nodetree add new empty node tree, set fakeuser immediately ''' if not context.space_data.tree_type in {'SverchCustomTreeType', }: print('not running from a sv nodetree') return if not hasattr(context.space_data.edit_tree, 'nodes'): msg_one = 'going to add a new empty node tree' msg_two = 'added new node tree' print(msg_one) operator.report({"WARNING"}, msg_one) ng_params = {'name': 'NodeTree', 'type': 'SverchCustomTreeType'} ng = bpy.data.node_groups.new(**ng_params) ng.use_fake_user = True context.space_data.node_tree = ng operator.report({"WARNING"}, msg_two)
- def verbose_macro_handler(operator, context, term)
- 
Expand source code@classmethod def verbose_macro_handler(cls, operator, context, term): cls.ensure_nodetree(operator, context) tree = context.space_data.edit_tree nodes, links = tree.nodes, tree.links if term == 'obj vd': obj_in_node = nodes.new('SvObjInLite') obj_in_node.dget() vd_node = nodes.new('SvViewerDrawMk4') vd_node.location = obj_in_node.location.x + 180, obj_in_node.location.y links.new(obj_in_node.outputs[0], vd_node.inputs[0]) links.new(obj_in_node.outputs[2], vd_node.inputs[2]) links.new(obj_in_node.outputs[4], vd_node.inputs[3]) elif term == 'objs vd': obj_in_node = nodes.new('SvGetObjectsData') obj_in_node.get_objects_from_scene(operator) vd_node = nodes.new('SvViewerDrawMk4') vd_node.location = obj_in_node.location.x + 180, obj_in_node.location.y # this macro could detect specifically if the node found edges or faces or both... links.new(obj_in_node.outputs[0], vd_node.inputs[0]) links.new(obj_in_node.outputs[2], vd_node.inputs[2]) links.new(obj_in_node.outputs[8], vd_node.inputs[3]) elif term == "objs socket to data": objdata_macro_one(context, operator, term, nodes, links) elif term == "3dcursor_to_vector_in": MOUSE_X, MOUSE_Y = context.space_data.cursor_location cursor = context.scene.cursor.location node = nodes.new("GenVectorsNode") node.location = MOUSE_X, MOUSE_Y node.x_, node.y_, node.z_ = tuple(cursor) elif term == "3dcursor_to_matrix": def flattened(matrix): return list(matrix[0]) + list(matrix[1]) + list(matrix[2]) + list(matrix[3]) MOUSE_X, MOUSE_Y = context.space_data.cursor_location matrix = context.scene.cursor.matrix node = nodes.new("SvMatrixValueIn") # "SvMatrixInNodeMK4") node.location = MOUSE_X, MOUSE_Y node.matrix = flattened(matrix.transposed()) elif term == 'rndcol': # all locations will be relative to cursor MOUSE_X, MOUSE_Y = context.space_data.cursor_location # add nodes to layout NUM = nodes.new("SvNumberNode") RR = nodes.new('NodeReroute') RND_0 = nodes.new('SvRndNumGen') RND_1 = nodes.new('SvRndNumGen') RND_2 = nodes.new('SvRndNumGen') COL = nodes.new('SvColorsInNodeMK1') # set locations COL.location = MOUSE_X + 140, MOUSE_Y + 40 RND_0.location.x = MOUSE_X - 40 RND_0.location.y = MOUSE_Y + 40 RND_1.location.x = MOUSE_X - 40 RND_1.location.y = MOUSE_Y RND_2.location.x = MOUSE_X - 40 RND_2.location.y = MOUSE_Y - 40 RR.location = MOUSE_X - 110, MOUSE_Y NUM.location = MOUSE_X - 280, MOUSE_Y + 20 # configure nodes set_node_props(NUM, { 'selected_mode': 'int', 'int_': 30}) set_node_props(RND_0, { 'type_selected_mode': 'Float', 'seed': 20, 'hide': True}) set_node_props(RND_1, { 'type_selected_mode': 'Float', 'seed': 30, 'hide': True}) set_node_props(RND_2, { 'type_selected_mode': 'Float', 'seed': 40, 'hide': True}) # link nodes links.new(NUM.outputs[0], RR.inputs[0]) links.new(RR.outputs[0], RND_0.inputs[0]) links.new(RR.outputs[0], RND_1.inputs[0]) links.new(RR.outputs[0], RND_2.inputs[0]) links.new(RND_0.outputs[0], COL.inputs[0]) links.new(RND_1.outputs[0], COL.inputs[1]) links.new(RND_2.outputs[0], COL.inputs[2]) elif 'hotswap' in term: swap_vd_mv(context, operator, term, nodes, links) return elif term == 'nuke python++': bpy.ops.script.reload() elif term == 'zen': full_url_term = 'https://gist.github.com/zeffii/d843b985b0db97af56dfa9c30cd54712' webbrowser.open(full_url_term) elif 'snl' in term: file = term.split(' ')[1] snlite = nodes.new('SvScriptNodeLite') snlite.location = context.space_data.cursor_location sn_loader(snlite, script_name=file) elif "join" in term: join_macros(context, operator, term, nodes, links) elif "math" in term: math_macros(context, operator, term, nodes, links) return elif "switch" in term: switch_macros(context, operator, term, nodes, links) return elif 'output numpy' in term: state_ = term.split(' ')[2] state = state_ == 'True' for node in nodes: # most of cases if hasattr(node, 'output_numpy'): node.output_numpy = state # line node if hasattr(node, 'as_numpy'): node.as_numpy = state # box node if hasattr(node, 'out_np'): for i in range(len(node.out_np)): node.out_np[i] = state # vector in if hasattr(node, 'implementation'): try: node.implementation = 'NumPy' if state else 'Python' except TypeError: pass elif term == 'gp +': gp_macro_one(context, operator, term, nodes, links) elif term == 'gp + 2': gp_macro_two(context, operator, term, nodes, links) elif term == 'url': bpy.ops.node.sv_load_archived_blend_url() elif term == 'blend 2 zip': bpy.ops.node.blend_to_archive(archive_ext="zip")
 
- class SvLoadArchivedBlendURL (...)
- 
Expand source codeclass SvLoadArchivedBlendURL(bpy.types.Operator): bl_idname = "node.sv_load_archived_blend_url" bl_label = "Load archive URL (.zip .gz of blend)" """ loads links like: https://github.com/nortikin/sverchok/files/647412/blend_name_YYYY_MM_DD_HH_mm.zip https://github.com/nortikin/sverchok/files/647412/blend_name_YYYY_MM_DD_HH_mm.gz """ download_url: bpy.props.StringProperty() os_temp_path = tempfile.gettempdir() def execute(self, context): wm = bpy.data.window_managers[0] wm.progress_begin(0, 100) wm.progress_update(20) if not self.download_url: clipboard = context.window_manager.clipboard if not clipboard: self.report({'ERROR'}, "Clipboard empty") return {'CANCELLED'} else: self.download_url = clipboard # get the file and obtain the new file's path on disk try: file_and_ext = os.path.basename(self.download_url) to_path = os.path.join(self.os_temp_path, file_and_ext) file = urllib.request.urlretrieve(self.download_url, to_path) wm.progress_update(50) except Exception as fullerr: print(repr(fullerr)) self.report({'ERROR'}, "Cannot get archive from Internet") wm.progress_end() return {'CANCELLED'} if file_and_ext.endswith('.gz'): return handle_gz(self, wm, to_path) elif file_and_ext.endswith('.zip'): return handle_zip(self, wm, to_path, file) else: # maybe if the url ends in .blend we could load it anyway. self.report({'ERROR'}, "url is not a .gz or .zip... ending operator") wm.progress_end() return {'CANCELLED'} return {'FINISHED'}Ancestors- bpy_types.Operator
- builtins.bpy_struct
 Class variables- var bl_idname
- var bl_label
- var bl_rna
- var download_url : <_PropertyDeferred,- , {'attr': 'download_url'}> 
- var os_temp_path
 Methods- def execute(self, context)
- 
Expand source codedef execute(self, context): wm = bpy.data.window_managers[0] wm.progress_begin(0, 100) wm.progress_update(20) if not self.download_url: clipboard = context.window_manager.clipboard if not clipboard: self.report({'ERROR'}, "Clipboard empty") return {'CANCELLED'} else: self.download_url = clipboard # get the file and obtain the new file's path on disk try: file_and_ext = os.path.basename(self.download_url) to_path = os.path.join(self.os_temp_path, file_and_ext) file = urllib.request.urlretrieve(self.download_url, to_path) wm.progress_update(50) except Exception as fullerr: print(repr(fullerr)) self.report({'ERROR'}, "Cannot get archive from Internet") wm.progress_end() return {'CANCELLED'} if file_and_ext.endswith('.gz'): return handle_gz(self, wm, to_path) elif file_and_ext.endswith('.zip'): return handle_zip(self, wm, to_path, file) else: # maybe if the url ends in .blend we could load it anyway. self.report({'ERROR'}, "url is not a .gz or .zip... ending operator") wm.progress_end() return {'CANCELLED'} return {'FINISHED'}