Source code for cam.parametric

"""Fabex 'parametric.py' © 2019 Devon (Gorialis) R

MIT License

Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:

The above copyright notice and this permission notice shall be included in all
copies or substantial portions of the Software.

THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE

Summary:
Create a Blender curve from a 3D parametric function.
This allows for a 3D plot to be made of the function, which can be converted into a mesh.

I have documented the inner workings here, but if you're not interested and just want to
suit this to your own function, scroll down to the bottom and edit the `f(t)` function and
the iteration count to your liking.

This code has been checked to work on Blender 2.92.
"""

from math import pow

import bpy
from mathutils import Vector


[docs] def derive_bezier_handles(a, b, c, d, tb, tc): """ Derives bezier handles by using the start and end of the curve with 2 intermediate points to use for interpolation. :param a: The start point. :param b: The first mid-point, located at `tb` on the bezier segment, where 0 < `tb` < 1. :param c: The second mid-point, located at `tc` on the bezier segment, where 0 < `tc` < 1. :param d: The end point. :param tb: The position of the first point in the bezier segment. :param tc: The position of the second point in the bezier segment. :return: A tuple of the two intermediate handles, that is, the right handle of the start point and the left handle of the end point. """ # Calculate matrix coefficients matrix_a = 3 * pow(1 - tb, 2) * tb matrix_b = 3 * (1 - tb) * pow(tb, 2) matrix_c = 3 * pow(1 - tc, 2) * tc matrix_d = 3 * (1 - tc) * pow(tc, 2) # Calculate the matrix determinant matrix_determinant = 1 / ((matrix_a * matrix_d) - (matrix_b * matrix_c)) # Calculate the components of the target position vector final_b = b - (pow(1 - tb, 3) * a) - (pow(tb, 3) * d) final_c = c - (pow(1 - tc, 3) * a) - (pow(tc, 3) * d) # Multiply the inversed matrix with the position vector to get the handle points bezier_b = matrix_determinant * ((matrix_d * final_b) + (-matrix_b * final_c)) bezier_c = matrix_determinant * ((-matrix_c * final_b) + (matrix_a * final_c)) # Return the handle points return bezier_b, bezier_c
[docs] def create_parametric_curve( function, *args, min: float = 0.0, max: float = 1.0, use_cubic: bool = True, iterations: int = 8, resolution_u: int = 10, **kwargs, ): """ Creates a Blender bezier curve object from a parametric function. This "plots" the function in 3D space from `min <= t <= max`. :param function: The function to plot as a Blender curve. This function should take in a float value of `t` and return a 3-item tuple or list of the X, Y and Z coordinates at that point: `function(t) -> (x, y, z)` `t` is plotted according to `min <= t <= max`, but if `use_cubic` is enabled, this function needs to be able to take values less than `min` and greater than `max`. :param *args: Additional positional arguments to be passed to the plotting function. These are not required. :param use_cubic: Whether or not to calculate the cubic bezier handles as to create smoother splines. Turning this off reduces calculation time and memory usage, but produces more jagged splines with sharp edges. :param iterations: The 'subdivisions' of the parametric to plot. Setting this higher produces more accurate curves but increases calculation time and memory usage. :param resolution_u: The preview surface resolution in the U direction of the bezier curve. Setting this to a higher value produces smoother curves in rendering, and increases the number of vertices the curve will get if converted into a mesh (e.g. for edge looping) :param **kwargs: Additional keyword arguments to be passed to the plotting function. These are not required. :return: The new Blender object. """ # Create the Curve to populate with points. curve = bpy.data.curves.new("Parametric", type="CURVE") curve.dimensions = "3D" curve.resolution_u = 30 # Add a new spline and give it the appropriate amount of points spline = curve.splines.new("BEZIER") spline.bezier_points.add(iterations) if use_cubic: points = [ function(((i - 3) / (3 * iterations)) * (max - min) + min, *args, **kwargs) for i in range((3 * (iterations + 2)) + 1) ] # Convert intermediate points into handles for i in range(iterations + 2): a = points[(3 * i)] b = points[(3 * i) + 1] c = points[(3 * i) + 2] d = points[(3 * i) + 3] bezier_bx, bezier_cx = derive_bezier_handles(a[0], b[0], c[0], d[0], 1 / 3, 2 / 3) bezier_by, bezier_cy = derive_bezier_handles(a[1], b[1], c[1], d[1], 1 / 3, 2 / 3) bezier_bz, bezier_cz = derive_bezier_handles(a[2], b[2], c[2], d[2], 1 / 3, 2 / 3) points[(3 * i) + 1] = (bezier_bx, bezier_by, bezier_bz) points[(3 * i) + 2] = (bezier_cx, bezier_cy, bezier_cz) # Set point coordinates and handles for i in range(iterations + 1): spline.bezier_points[i].co = points[3 * (i + 1)] spline.bezier_points[i].handle_left_type = "FREE" spline.bezier_points[i].handle_left = Vector(points[(3 * (i + 1)) - 1]) spline.bezier_points[i].handle_right_type = "FREE" spline.bezier_points[i].handle_right = Vector(points[(3 * (i + 1)) + 1]) else: points = [function(i / iterations, *args, **kwargs) for i in range(iterations + 1)] # Set point coordinates, disable handles for i in range(iterations + 1): spline.bezier_points[i].co = points[i] spline.bezier_points[i].handle_left_type = "VECTOR" spline.bezier_points[i].handle_right_type = "VECTOR" # Create the Blender object and link it to the scene curve_object = bpy.data.objects.new("Parametric", curve) context = bpy.context scene = context.scene link_object = scene.collection.objects.link link_object(curve_object) # Return the new object return curve_object
[docs] def make_edge_loops(*objects): """ Turns a set of Curve objects into meshes, creates vertex groups, and merges them into a set of edge loops. :param *objects: Positional arguments for each object to be converted and merged. """ context = bpy.context scene = context.scene mesh_objects = [] vertex_groups = [] # Convert all curves to meshes for obj in objects: # Unlink old object unlink_object(obj) # Convert curve to a mesh if bpy.app.version >= (2, 80): new_mesh = obj.to_mesh().copy() else: new_mesh = obj.to_mesh(scene, False, "PREVIEW") # Store name and matrix, then fully delete the old object name = obj.name matrix = obj.matrix_world bpy.data.objects.remove(obj) # Attach the new mesh to a new object with the old name new_object = bpy.data.objects.new(name, new_mesh) new_object.matrix_world = matrix # Make a new vertex group from all vertices on this mesh vertex_group = new_object.vertex_groups.new(name=name) vertex_group.add(range(len(new_mesh.vertices)), 1.0, "ADD") vertex_groups.append(vertex_group) # Link our new object link_object(new_object) # Add it to our list mesh_objects.append(new_object) # Make a new context ctx = context.copy() # Select our objects in the context ctx["active_object"] = mesh_objects[0] ctx["selected_objects"] = mesh_objects if bpy.app.version >= (2, 80): ctx["selected_editable_objects"] = mesh_objects else: ctx["selected_editable_bases"] = [scene.object_bases[o.name] for o in mesh_objects] # Join them together bpy.ops.object.join(ctx)