-
-
Save zeffii/11306138 to your computer and use it in GitHub Desktop.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
from node_s import * | |
from util import * | |
from mathutils import Vector, Euler, kdtree | |
import bpy | |
from bpy.props import StringProperty | |
class SvKDTreeNode(Node, SverchCustomTreeNode): | |
bl_idname = 'SvKDTreeNode' | |
bl_label = 'Kdtree search' | |
bl_icon = 'OUTLINER_OB_EMPTY' | |
current_mode = StringProperty(default="FIND") | |
def mode_change(self, context): | |
# just because click doesn't mean we need to change mode | |
mode = self.mode | |
if mode == self.current_mode: | |
return | |
inputs = self.inputs | |
outputs = self.outputs | |
socket_map = { | |
'Verts': 'VerticesSocket', | |
'Check Verts': 'VerticesSocket', | |
'proximity Verts': 'VerticesSocket', | |
'proximity Indices': 'StringsSocket', | |
'proximity Dists': 'StringsSocket', | |
'radius': 'StringsSocket', | |
'n nearest': 'StringsSocket', | |
'n proxima .co': 'VerticesSocket', | |
'n proxima .idx': 'StringsSocket', | |
'n proxima dist': 'StringsSocket', | |
'grouped .co': 'VerticesSocket', | |
'grouped .idx': 'StringsSocket', | |
'grouped dist': 'StringsSocket' | |
} | |
standard_inputs = ['Verts', 'Check Verts'] | |
while len(inputs) > 2: | |
inputs.remove(inputs[-1]) | |
while len(outputs) > 0: | |
outputs.remove(outputs[-1]) | |
if mode == 'FIND': | |
self.current_mode = mode | |
outs = ['proximity Verts', 'proximity Indices', 'proximity Dists'] | |
for socket_name in outs: | |
socket_type = socket_map[socket_name] | |
outputs.new(socket_type, socket_name, socket_name) | |
elif mode == 'FIND_N': | |
self.current_mode = mode | |
socket_name = 'n nearest' | |
socket_type = socket_map[socket_name] | |
inputs.new(socket_type, socket_name, socket_name) | |
outs = ['n proxima .co', 'n proxima .idx', 'n proxima dist'] | |
for socket_name in outs: | |
socket_type = socket_map[socket_name] | |
outputs.new(socket_type, socket_name, socket_name) | |
elif mode == 'FIND_RANGE': | |
self.current_mode = mode | |
socket_name = 'radius' | |
socket_type = socket_map[socket_name] | |
inputs.new(socket_type, socket_name, socket_name) | |
outs = ['grouped .co', 'grouped .idx', 'grouped dist'] | |
for socket_name in outs: | |
socket_type = socket_map[socket_name] | |
outputs.new(socket_type, socket_name, socket_name) | |
else: | |
return | |
modes = [ | |
("FIND", " 1 ", "Find nearest", 1), | |
("FIND_N", " n ", "Find n nearest", 2), | |
("FIND_RANGE", "radius", "Find within Distance", 3) | |
] | |
mode = bpy.props.EnumProperty( | |
items=modes, default='FIND', update=mode_change) | |
def draw_buttons(self, context, layout): | |
layout.label("Search mode:") | |
layout.prop(self, "mode", expand=True) | |
def init(self, context): | |
self.inputs.new('VerticesSocket', 'Verts', 'Verts') | |
self.inputs.new('VerticesSocket', 'Check Verts', 'Check Verts') | |
defaults = ['proximity Verts', 'proximity Indices', 'proximity Dists'] | |
pVerts, pIdxs, pDists = defaults | |
self.outputs.new('VerticesSocket', pVerts, pVerts) | |
self.outputs.new('StringsSocket', pIdxs, pIdxs) | |
self.outputs.new('StringsSocket', pDists, pDists) | |
def update(self): | |
inputs = self.inputs | |
outputs = self.outputs | |
if not ('Verts' in inputs and inputs['Verts'].links): | |
return | |
try: | |
verts = SvGetSocketAnyType(self, inputs['Verts'])[0] | |
except IndexError: | |
return | |
''' | |
- assumptions: | |
: MainVerts are potentially different on each update | |
: not nested input ([vlist1],[vlist2],...) | |
with small vert lists I don't imagine this will be very noticeable, | |
''' | |
# make kdtree | |
# documentation/blender_python_api_2_70_release/mathutils.kdtree.html | |
size = len(verts) | |
kd = mathutils.kdtree.KDTree(size) | |
for i, vtx in enumerate(verts): | |
kd.insert(Vector(vtx), i) | |
kd.balance() | |
reset_outs = { | |
'FIND': [ | |
'proximity Verts', 'proximity Indices', 'proximity Dists'], | |
'FIND_N': [ | |
'n proxima .co', 'n proxima .idx', 'n proxima dist'], | |
'FIND_RANGE': [ | |
'grouped .co', 'grouped .idx', 'grouped dist'] | |
} | |
# set sockets to [] and return early, somehow this has been triggered | |
# early. | |
if not ('Check Verts' in inputs and inputs['Check Verts']): | |
try: | |
for socket in reset_outs[self.mode]: | |
SvSetSocketAnyType(self, socket, []) | |
except KeyError: | |
pass | |
return | |
# before continuing check at least that there is one output. | |
try: | |
some_output = any([outputs[i].links for i in range(3)]) | |
except (IndexError, KeyError) as e: | |
return | |
if self.mode == 'FIND': | |
''' [Verts.co,..] => | |
[Verts.idx,.] => | |
[Verts.dist,.] => | |
=> [Main Verts] | |
=> [cVert,..] | |
''' | |
try: | |
cVerts = SvGetSocketAnyType(self, inputs['Check Verts'])[0] | |
except (IndexError, KeyError) as e: | |
return | |
verts_co_list = [] | |
verts_idx_list = [] | |
verts_dist_list = [] | |
add_verts_coords = verts_co_list.append | |
add_verts_idxs = verts_idx_list.append | |
add_verts_dists = verts_dist_list.append | |
for i, vtx in enumerate(cVerts): | |
co, index, dist = kd.find(vtx) | |
add_verts_coords([co.to_tuple()]) | |
add_verts_idxs(index) | |
add_verts_dists(dist) | |
SvSetSocketAnyType(self, 'proximity Verts', verts_co_list) | |
SvSetSocketAnyType(self, 'proximity Indices', verts_idx_list) | |
SvSetSocketAnyType(self, 'proximity Dists', verts_dist_list) | |
elif self.mode == 'FIND_N': | |
''' [[Verts.co,..n],..c] => from MainVerts closest to v.co | |
[[Verts.idx,..n],.c] => from MainVerts closest to v.co | |
[[Verts.dist,.n],.c] => from MainVerts closest to v.co | |
=> [Main Verts] | |
=> [cVert,..] | |
=> [n, max n nearest | |
''' | |
try: | |
cVerts = SvGetSocketAnyType(self, inputs['Check Verts'])[0] | |
n = SvGetSocketAnyType(self, inputs['n nearest'])[0][0] | |
except (IndexError, KeyError) as e: | |
return | |
if n < 1: | |
return | |
n_proxima_co = [] | |
n_proxima_idx = [] | |
n_proxima_dist = [] | |
add_co_proximas = n_proxima_co.append | |
add_idx_proximas = n_proxima_idx.append | |
add_dist_proximas = n_proxima_dist.append | |
for i, vtx in enumerate(cVerts): | |
co_list = [] | |
idx_list = [] | |
dist_list = [] | |
n_list = kd.find_n(vtx, n) | |
for co, index, dist in n_list: | |
co_list.append([co.to_tuple()]) | |
idx_list.append(index) | |
dist_list.append(dist) | |
add_co_proximas(co_list) | |
add_idx_proximas(idx_list) | |
add_dist_proximas(dist_list) | |
SvSetSocketAnyType(self, 'n proxima .co', n_proxima_co) | |
SvSetSocketAnyType(self, 'n proxima .idx', n_proxima_idx) | |
SvSetSocketAnyType(self, 'n proxima dist', n_proxima_dist) | |
elif self.mode == 'FIND_RANGE': | |
''' [grouped [.co for p in MainVerts in r of v in cVert]] => | |
[grouped [.idx for p in MainVerts in r of v in cVert]] => | |
[grouped [.dist for p in MainVerts in r of v in cVert]] => | |
=> [Main Verts] | |
=> [cVert,..] | |
=> n | |
''' | |
try: | |
cVerts = SvGetSocketAnyType(self, inputs['Check Verts'])[0] | |
n = SvGetSocketAnyType(self, inputs['radius'])[0][0] | |
except (IndexError, KeyError) as e: | |
return | |
# for i, vtx in enumerate(verts): | |
# num_edges = 0 | |
# for (co, index, dist) in kd.find_range(vtx, mdist): | |
# if i == index or (num_edges > 2): | |
# continue | |
# e.append([i, index]) | |
# num_edges += 1 | |
pass | |
def update_socket(self, context): | |
self.update() | |
def register(): | |
bpy.utils.register_class(SvKDTreeNode) | |
def unregister(): | |
bpy.utils.unregister_class(SvKDTreeNode) | |
if __name__ == "__main__": | |
register() |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment