Skip to content
Snippets Groups Projects
object_grease_scatter.py 13.5 KiB
Newer Older
# ##### 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": "Grease Scatter Objects",
    "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, Matrix, Quaternion
from random import uniform, shuffle
import bpy

Campbell Barton's avatar
Campbell Barton committed

def _main(self, DENSITY=1.0, SCALE=0.6, RAND_LOC=0.8, RAND_ALIGN=0.75):
    from math import radians
Campbell Barton's avatar
Campbell Barton committed

    C = bpy.context
    o = C.object
    # print(o.ray_cast(Vector(), Vector(0,0,0.2)))
Campbell Barton's avatar
Campbell Barton committed

Campbell Barton's avatar
Campbell Barton committed
    SEEK = 2.0  # distance for ray to seek
    BAD_NORMAL = Vector((0.0, 0.0, -1.0))
Campbell Barton's avatar
Campbell Barton committed
    mats = [Matrix.Rotation(radians(-45), 3, 'X'),
            Matrix.Rotation(radians(+45), 3, 'X'),
            Matrix.Rotation(radians(-45), 3, 'Y'),
            Matrix.Rotation(radians(+45), 3, 'Y'),
            Matrix.Rotation(radians(-45), 3, 'Z'),
            Matrix.Rotation(radians(+45), 3, 'Z'),
            ]

    Z_UP = Vector((0.0, 0.0, 1.0))
    dirs = [Vector((0.0, 0.0, OFS)),
            Vector((0.0, 0.0, -OFS)),
            ]
    '''
    Vector(0,OFS,0),
    Vector(0,-OFS,0),
    Vector(OFS,0,0),
    Vector(-OFS,0,0)
    '''
Campbell Barton's avatar
Campbell Barton committed

    group = bpy.data.groups.get(o.name)
Campbell Barton's avatar
Campbell Barton committed

    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

Campbell Barton's avatar
Campbell Barton committed
    def debug_edge(v1, v2):
        mesh = bpy.data.meshes.new("Retopo")
Campbell Barton's avatar
Campbell Barton committed
        mesh.from_pydata([v1, v2], [(0.0, 1.0)], [])

        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
Campbell Barton's avatar
Campbell Barton committed

    def fix_point(p):
        for d in dirs:
            # print(p)
            hit, no, ind = ray(p, p + d)
            if ind != -1:
                if DEBUG:
Campbell Barton's avatar
Campbell Barton committed
                    return [p, no, None]
Campbell Barton's avatar
Campbell Barton committed
                    return [hit, no, None]
Campbell Barton's avatar
Campbell Barton committed

        return [p, BAD_NORMAL, None]

    def get_points(stroke):
        return [fix_point(point.co) for point in stroke.points]

    def get_splines(gp):
        if gp.layers.active:
            frame = gp.layers.active.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
Campbell Barton's avatar
Campbell Barton committed

        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
Campbell Barton's avatar
Campbell Barton committed

                # # dont self intersect
                best_nor = None
Campbell Barton's avatar
Campbell Barton committed
                #best_hit = None
                best_dist = 10000000.0
                pofs = p + n * 0.01
Campbell Barton's avatar
Campbell Barton committed

                m_alt_1 = Matrix.Rotation(radians(22.5), 3, n)
                m_alt_2 = Matrix.Rotation(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 + (n_seek * m))
                        if ind != -1:
                            dist = (pofs - hit).length
                            if dist < best_dist:
                                best_dist = dist
                                best_nor = nor
                                #best_hit = hit
Campbell Barton's avatar
Campbell Barton committed

                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
Campbell Barton's avatar
Campbell Barton committed

        # Now we need to do scattering.
        # first corners
        hits = []
        nors = []
        oris = []
        for s in splines:
Campbell Barton's avatar
Campbell Barton committed
            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
Campbell Barton's avatar
Campbell Barton committed

                    vantage = p + n
                    if DEBUG:
                        p[:] = vantage
Campbell Barton's avatar
Campbell Barton committed

                    # 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)):
Campbell Barton's avatar
Campbell Barton committed
                    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
                        '''
Campbell Barton's avatar
Campbell Barton committed

                        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
Campbell Barton's avatar
Campbell Barton committed

                        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)
Campbell Barton's avatar
Campbell Barton committed

        if 0:
            mesh = bpy.data.meshes.new("Retopo")
            mesh.from_pydata(hits, [], [])
Campbell Barton's avatar
Campbell Barton committed

            scene = bpy.context.scene
            mesh.update()
            obj_new = bpy.data.objects.new("Torus", mesh)
            scene.objects.link(obj_new)
            obj_new.layers[:] = o.layers
Campbell Barton's avatar
Campbell Barton committed

            # 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:
Campbell Barton's avatar
Campbell Barton committed

            def apply_faces(triples):
                # first randomize the faces
                shuffle(triples)
Campbell Barton's avatar
Campbell Barton committed

                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] = []
Campbell Barton's avatar
Campbell Barton committed

                    vv = [tuple(v) for f in triple_subset for v in f]

                    mesh = bpy.data.meshes.new("Retopo")
Campbell Barton's avatar
Campbell Barton committed
                    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
Campbell Barton's avatar
Campbell Barton committed

                    # Now setup dupli-faces
                    obj_new.dupli_type = 'FACES'
                    obj_new.use_dupli_faces_scale = True
                    obj_new.dupli_faces_scale = 100.0
Campbell Barton's avatar
Campbell Barton committed

                    inst_ob.location = obj_new.location
                    inst_ob.parent = obj_new
Campbell Barton's avatar
Campbell Barton committed

                    # 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
                    '''
Campbell Barton's avatar
Campbell Barton committed

            # build faces from vert/normals
Campbell Barton's avatar
Campbell Barton committed
            tri = (Vector((0.0, 0.0, 0.01)),
                   Vector((0.0, 0.0, 0.0)),
                   Vector((0.0, 0.01, 0.01)))

Campbell Barton's avatar
Campbell Barton committed
            # face_ind = []
            for i in range(len(hits)):
                co = hits[i]
                no = nors[i]
                ori = oris[i]
                quat = no.to_track_quat('X', 'Z')
Campbell Barton's avatar
Campbell Barton committed

Campbell Barton's avatar
Campbell Barton committed
                angle = radians(uniform(-180, 180.0))
                angle_aligned = -(ori.angle(Vector((0.0, 1.0, 0.0)) * quat, radians(180.0)))

                quat = Quaternion(no, (angle * (1.0 - RAND_ALIGN)) + (angle_aligned * RAND_ALIGN)).cross(quat)
Campbell Barton's avatar
Campbell Barton committed

                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])
Campbell Barton's avatar
Campbell Barton committed

Campbell Barton's avatar
Campbell Barton committed
from bpy.props import FloatProperty


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)
Campbell Barton's avatar
Campbell Barton committed

    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:
Campbell Barton's avatar
Campbell Barton committed
                    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.window_manager
        wm.invoke_popup(self, width=180)
        return {'RUNNING_MODAL'}

    def draw(self, context):
        self.__class__._parent = self
        layout = self.layout
Campbell Barton's avatar
Campbell Barton committed

        for attr in self.__class__.__dict__["order"]:
            if not attr.startswith("_"):
                try:
                    layout.prop(self.properties, attr)
                except:
                    pass

        layout.operator_context = 'EXEC_DEFAULT'
Campbell Barton's avatar
Campbell Barton committed
        layout.operator(self.bl_idname)


# Add to the menu
menu_func = (lambda self, context: self.layout.operator(Scatter.bl_idname,
                                        text="Scatter", icon='AUTO'))

Campbell Barton's avatar
Campbell Barton committed

    bpy.utils.register_class(Scatter)
    bpy.types.VIEW3D_PT_tools_objectmode.append(menu_func)

Campbell Barton's avatar
Campbell Barton committed

    bpy.utils.unregister_class(Scatter)
    bpy.types.VIEW3D_PT_tools_objectmode.remove(menu_func)

#if __name__ == "__main__":
#    _main()