From c556bd0b7879722d4a105979a391afba74ddd1d8 Mon Sep 17 00:00:00 2001 From: Campbell Barton <ideasman42@gmail.com> Date: Sun, 17 Jul 2011 13:03:05 +0000 Subject: [PATCH] initial commit of grease scatter script, still needs updating. --- object_grease_scatter.py | 426 +++++++++++++++++++++++++++++++++++++++ 1 file changed, 426 insertions(+) create mode 100644 object_grease_scatter.py diff --git a/object_grease_scatter.py b/object_grease_scatter.py new file mode 100644 index 000000000..32be930c9 --- /dev/null +++ b/object_grease_scatter.py @@ -0,0 +1,426 @@ +# ##### 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 ##### + +# <pep8 compliant> + +# Script copyright (C) Campbell Barton + +bl_info = { + "name": "Export Camera Animation", + "author": "Campbell Barton", + "version": (0, 1), + "blender": (2, 5, 8), + "api": 36079, + "location": "File > Export > Cameras & Markers (.py)", + "description": "Export Cameras & Markers (.py)", + "warning": "", + "wiki_url": "http://wiki.blender.org/index.php/Extensions:2.5/Py/"\ + "Scripts/Object/Grease_Scatter", + "tracker_url": "https://projects.blender.org/tracker/index.php?"\ + "func=detail&aid=TODO", + "support": 'OFFICIAL', + "category": "Object"} + +from mathutils import Vector, RotationMatrix, Quaternion +from math import radians +from random import uniform, shuffle +import bpy + +def _main(self, DENSITY=1.0, SCALE=0.6, RAND_LOC=0.8, RAND_ALIGN=0.75): + from math import radians + + C = bpy.context + o = C.object + # print(o.ray_cast(Vector(), Vector(0,0,0.2))) + + OFS = 0.2 + SEEK = 2.0 # distance for ray to seek + BAD_NORMAL = Vector((0,0,-1)) + WALL_LIMIT = radians(45.0) + + mats = [ + RotationMatrix(radians(-45), 3, 'X'), + RotationMatrix(radians( 45), 3, 'X'), + RotationMatrix(radians(-45), 3, 'Y'), + RotationMatrix(radians( 45), 3, 'Y'), + RotationMatrix(radians(-45), 3, 'Z'), + RotationMatrix(radians( 45), 3, 'Z')] + + + Z_UP = Vector((0,0,1.0)) + dirs = [ + Vector((0,0,OFS)), + Vector((0,0,-OFS))] + ''' + Vector(0,OFS,0), + Vector(0,-OFS,0), + Vector(OFS,0,0), + Vector(-OFS,0,0) + ''' + + group = bpy.data.groups.get(o.name) + + if not group: + self.report({'WARNING'}, "Group '%s' not found, must match object name" % o.name) + return + + def faces_from_hits(hit_list): + def from_pydata(self, verts, edges, faces): + """ + Make a mesh from a list of verts/edges/faces + Until we have a nicer way to make geometry, use this. + """ + self.add_geometry(len(verts), len(edges), len(faces)) + + verts_flat = [f for v in verts for f in v] + self.verts.foreach_set("co", verts_flat) + del verts_flat + + edges_flat = [i for e in edges for i in e] + self.edges.foreach_set("verts", edges_flat) + del edges_flat + + def treat_face(f): + if len(f) == 3: + return f[0], f[1], f[2], 0 + elif f[3] == 0: + return f[3], f[0], f[1], f[2] + return f + + faces_flat = [v for f in faces for v in treat_face(f)] + self.faces.foreach_set("verts_raw", faces_flat) + del faces_flat + + + + + def debug_edge(v1,v2): + mesh = bpy.data.meshes.new("Retopo") + mesh.from_pydata([v1,v2], [(0,1)], []) + + scene = bpy.context.scene + mesh.update() + obj_new = bpy.data.objects.new("Torus", mesh) + scene.objects.link(obj_new) + + ray = o.ray_cast + #ray = C.scene.ray_cast + + DEBUG = False + def fix_point(p): + for d in dirs: + # print(p) + hit, no, ind = ray(p, p + d) + if ind != -1: + if DEBUG: + return [p, no, None] + else: + # print("good", hit, no) + return [hit, no, None] + + # worry! + print("bad!", p, BAD_NORMAL) + + return [p, BAD_NORMAL, None] + + def get_points(stroke): + return [fix_point(point.co) for point in stroke.points] + + def get_splines(gp): + l = None + for l in gp.layers: + if l.active: # XXX - should be layers.active + break + if l: + frame = l.active_frame + return [get_points(stroke) for stroke in frame.strokes] + else: + return [] + + def main(): + scene = bpy.context.scene + obj = bpy.context.object + + gp = None + + if obj: + gp = obj.grease_pencil + + if not gp: + gp = scene.grease_pencil + + if not gp: + self.report({'WARNING'}, "No grease pencil layer found") + return + + splines = get_splines(gp) + + for s in splines: + for pt in s: + p = pt[0] + n = pt[1] + # print(p, n) + if n is BAD_NORMAL: + continue + + # # dont self intersect + best_nor = None + best_hit = None + best_dist = 10000000.0 + pofs = p + n * 0.01 + + n_seek = n * SEEK + m_alt_1 = RotationMatrix(radians(22.5), 3, n) + m_alt_2 = RotationMatrix(radians(-22.5), 3, n) + for _m in mats: + for m in (_m, m_alt_1 * _m, m_alt_2 * _m): + hit, nor, ind = ray(pofs, pofs + (m * n_seek)) + if ind != -1: + dist = (pofs - hit).length + if dist < best_dist: + best_dist = dist + best_nor = nor + #best_hit = hit + + if best_nor: + pt[1].length = best_dist + best_nor.negate() + pt[2] = best_nor + + + #scene.cursor_location[:] = best_hitnyway + # bpy.ops.wm.redraw_timer(type='DRAW_WIN_SWAP', iterations=1) + # debug_edge(p, best_hit) + # p[:] = best_hit + + # Now we need to do scattering. + # first corners + hits = [] + nors = [] + oris = [] + for s in splines: + for p,n,n_other in s: # point, normal, n_other the closest hit normal + if n is BAD_NORMAL: + continue + if n_other: + # cast vectors twice as long as the distance needed just incase. + n_down = (n * -SEEK) + l = n_down.length + n_other.length = l + + vantage = p + n + if DEBUG: + p[:] = vantage + + # We should cast rays between n_down and n_other + #for f in (0.0, 0.2, 0.4, 0.6, 0.8, 1.0): + TOT = int(10 * DENSITY) + #for i in list(range(TOT)): + for i in list(range(TOT))[int(TOT/1.5):]: # second half + f = i/(TOT-1) + + # focus on the center + ''' + f -= 0.5 + f = f*f + f += 0.5 + ''' + + ntmp = f * n_down + (1.0 - f)*n_other + # randomize + ntmp.x += uniform(-l, l) * RAND_LOC + ntmp.y += uniform(-l, l) * RAND_LOC + ntmp.z += uniform(-l, l) * RAND_LOC + + hit, hit_no, ind = ray(vantage, vantage + ntmp) + # print(hit, hit_no) + if ind != -1: + if hit_no.angle(Z_UP) < WALL_LIMIT: + hits.append(hit) + nors.append(hit_no) + oris.append(n_other.cross(hit_no)) + #oris.append(n_other) + + + + if 0: + mesh = bpy.data.meshes.new("Retopo") + mesh.from_pydata(hits, [], []) + + scene = bpy.context.scene + mesh.update() + obj_new = bpy.data.objects.new("Torus", mesh) + scene.objects.link(obj_new) + obj_new.layers[:] = o.layers + + # Now setup dupli-faces + obj_new.dupli_type = 'VERTS' + ob_child = bpy.data.objects["trash"] + ob_child.location = obj_new.location + ob_child.parent = obj_new + else: + + def apply_faces(triples): + # first randomize the faces + shuffle(triples) + + obs = group.objects[:] + tot = len(obs) + tot_div = int(len(triples) / tot) + + for inst_ob in obs: + triple_subset = triples[0:tot_div] + triples[0:tot_div] = [] + + vv = [tuple(v) for f in triple_subset for v in f] + + mesh = bpy.data.meshes.new("Retopo") + mesh.from_pydata(vv, [], [(i*3, i*3+1, i*3+2) for i in range(len(triple_subset))]) + + scene = bpy.context.scene + mesh.update() + obj_new = bpy.data.objects.new("Torus", mesh) + scene.objects.link(obj_new) + obj_new.layers[:] = o.layers + + # Now setup dupli-faces + obj_new.dupli_type = 'FACES' + obj_new.use_dupli_faces_scale = True + obj_new.dupli_faces_scale = 100.0 + + inst_ob.location = obj_new.location + inst_ob.parent = obj_new + + # BGE settings for testiing + ''' + inst_ob.game.physics_type = 'RIGID_BODY' + inst_ob.game.use_collision_bounds = True + inst_ob.game.collision_bounds = 'TRIANGLE_MESH' + inst_ob.game.collision_margin = 0.1 + obj_new.select = True + ''' + + + # build faces from vert/normals + tri = Vector((0, 0 ,0.01)), Vector((0, 0, 0)), Vector((0.0, 0.01, 0.01)) + + coords = [] + face_ind = [] + for i in range(len(hits)): + co = hits[i] + no = nors[i] + ori = oris[i] + quat = no.to_track_quat('X', 'Z') + + # make 2 angles and blend + angle = radians(uniform(-180, 180.0)) + angle_aligned = -(ori.angle(quat * Vector((0,1,0)), radians(180.0))) + + quat = Quaternion(no, (angle * (1.0-RAND_ALIGN)) + (angle_aligned * RAND_ALIGN)).cross(quat) + + f = uniform(0.1, 1.2) * SCALE + + coords.append([co + ((tri[0] * f) * quat), co + ((tri[1] * f) * quat), co + ((tri[2] * f) * quat)]) + # face_ind.append([i*3, i*3+1, i*3+2]) + + + apply_faces(coords) + + + main() + + +from bpy.props import * +class Scatter(bpy.types.Operator): + '''''' + bl_idname = "object.scatter" + bl_label = "Scatter" + bl_options = {'REGISTER'} + + density = FloatProperty(name="Density", + description="Multiplier for the density of items", + default=1.0, min=0.01, max=10.0) + + scale = FloatProperty(name="Scale", + description="Size multiplier for duplifaces", + default=1.0, min=0.01, max=10.0) + + rand_align = FloatProperty(name="Random Align", + description="Randomize alignmet with the walls", + default=0.75, min=0.0, max=1.0) + + rand_loc = FloatProperty(name="Random Loc", + description="Randomize Placement", + default=0.75, min=0.0, max=1.0) + + _parent = None + + def execute(self, context): + #self.properties.density = self.__class__._parent.properties.density # XXX bad way to copy args. + #self.properties.scale = self.__class__._parent.properties.scale # XXX bad way to copy args. + + for attr in self.__class__.__dict__["order"]: + if not attr.startswith("_"): + try: + setattr(self.properties, attr, getattr(self.__class__._parent.properties, attr)) + except: + pass + + _main(self, + DENSITY=self.properties.density, + SCALE=self.properties.scale, + RAND_LOC=self.properties.rand_loc, + RAND_ALIGN=self.properties.rand_align + ) + return {'FINISHED'} + + def invoke(self, context, event): + wm = context.manager + wm.invoke_popup(self, width=180) + return {'RUNNING_MODAL'} + + def draw(self, context): + self.__class__._parent = self + layout = self.layout + + for attr in self.__class__.__dict__["order"]: + if not attr.startswith("_"): + try: + layout.prop(self.properties, attr) + except: + pass + + layout.operator_context = 'EXEC_DEFAULT' + props = layout.operator(self.bl_idname) + return {'RUNNING_MODAL'} + + +# Add to the menu +menu_func = (lambda self, context: self.layout.operator(Scatter.bl_idname, + text="Scatter", icon='AUTO')) + +def register(): + bpy.types.register(Scatter) + bpy.types.VIEW3D_PT_tools_objectmode.append(menu_func) + +def unregister(): + bpy.types.unregister(Scatter) + bpy.types.VIEW3D_PT_tools_objectmode.remove(menu_func) + +#if __name__ == "__main__": +# _main() -- GitLab