How to resolve the algorithm Catmull–Clark subdivision surface step by step in the Python programming language

Published on 12 May 2024 09:40 PM

How to resolve the algorithm Catmull–Clark subdivision surface step by step in the Python programming language

Table of Contents

Problem Statement

Implement the Catmull-Clark surface subdivision (description on Wikipedia), which is an algorithm that maps from a surface (described as a set of points and a set of polygons with vertices at those points) to another more refined surface. The resulting surface will always consist of a mesh of quadrilaterals. The process for computing the new locations of the points works as follows when the surface is free of holes: Then each face is replaced by new faces made with the new points, When there is a hole, we can detect it as follows: On the border of a hole the subdivision occurs as follows: For edges and vertices not next to a hole, the standard algorithm from above is used.

Let's start with the solution:

Step by Step solution about How to resolve the algorithm Catmull–Clark subdivision surface step by step in the Python programming language

The provided code implements the Catmull–Clark subdivision surface algorithm. The Catmull–Clark subdivision surface algorithm is a mathematical technique used in computer graphics to generate smooth surfaces from a set of polygonal input faces. It works by iteratively dividing each face into smaller faces and moving the vertices of those faces to new positions that are determined by the positions of the original vertices and the vertices of the adjacent faces.

The Python code you provided takes two arguments:

  1. A list of points, which represent the vertices of the input polygons.
  2. A list of faces, which are lists of the indices of the points that make up each face.

This code implements the following steps of the Catmull–Clark subdivision surface algorithm:

  1. Compute the average of the points of each face. This is done by computing the sum of the points of the face and dividing by the number of points. The result is a face point.
  2. Compute the center of each edge. This is done by computing the average of the two points that make up the edge. The result is an edge center.
  3. Compute the average of the edge centers of each point. This is done by computing the sum of the edge centers of the point and dividing by the number of edge centers. The result is an average edge center.
  4. Compute the average of the face points of each point. This is done by computing the sum of the face points of the point and dividing by the number of face points. The result is an average face point.
  5. Compute the new position of each point. The new position is computed by taking the average of the old position, the average edge center, and the average face point.
  6. Compute the new faces. The new faces are computed by connecting the new points in the same order as the old faces.

The code then repeats these steps for a specified number of iterations. After each iteration, the number of faces and the number of points increase. The resulting surface is a smooth subdivision surface.

The code finally plots the subdivided surface using the matplotlib library.

Source code in the python programming language

"""

Input and output are assumed to be in this form based on the talk
page for the task:

input_points = [
  [-1.0,  1.0,  1.0],
  [-1.0, -1.0,  1.0],
  [ 1.0, -1.0,  1.0],
  [ 1.0,  1.0,  1.0],
  [ 1.0, -1.0, -1.0],
  [ 1.0,  1.0, -1.0],
  [-1.0, -1.0, -1.0],
  [-1.0,  1.0, -1.0]
]

input_faces = [
  [0, 1, 2, 3],
  [3, 2, 4, 5],
  [5, 4, 6, 7],
  [7, 0, 3, 5],
  [7, 6, 1, 0],
  [6, 1, 2, 4],
]

So, the graph is a list of points and a list of faces.
Each face is a list of indexes into the points list.
   
"""

from mpl_toolkits.mplot3d import axes3d
import matplotlib.pyplot as plt
import numpy as np
import sys

def center_point(p1, p2):
    """ 
    returns a point in the center of the 
    segment ended by points p1 and p2
    """
    cp = []
    for i in range(3):
        cp.append((p1[i]+p2[i])/2)
        
    return cp
    
def sum_point(p1, p2):
    """ 
    adds points p1 and p2
    """
    sp = []
    for i in range(3):
        sp.append(p1[i]+p2[i])
        
    return sp

def div_point(p, d):
    """ 
    divide point p by d
    """
    sp = []
    for i in range(3):
        sp.append(p[i]/d)
        
    return sp
    
def mul_point(p, m):
    """ 
    multiply point p by m
    """
    sp = []
    for i in range(3):
        sp.append(p[i]*m)
        
    return sp

def get_face_points(input_points, input_faces):
    """
    From http://rosettacode.org/wiki/Catmull%E2%80%93Clark_subdivision_surface
    
    1. for each face, a face point is created which is the average of all the points of the face.
    """

    # 3 dimensional space
    
    NUM_DIMENSIONS = 3
    
    # face_points will have one point for each face
    
    face_points = []
    
    for curr_face in input_faces:
        face_point = [0.0, 0.0, 0.0]
        for curr_point_index in curr_face:
            curr_point = input_points[curr_point_index]
            # add curr_point to face_point
            # will divide later
            for i in range(NUM_DIMENSIONS):
                face_point[i] += curr_point[i]
        # divide by number of points for average
        num_points = len(curr_face)
        for i in range(NUM_DIMENSIONS):
            face_point[i] /= num_points
        face_points.append(face_point)
        
    return face_points
    
def get_edges_faces(input_points, input_faces):
    """
    
    Get list of edges and the one or two adjacent faces in a list.
    also get center point of edge
    
    Each edge would be [pointnum_1, pointnum_2, facenum_1, facenum_2, center]
    
    """
    
    # will have [pointnum_1, pointnum_2, facenum]
    
    edges = []
    
    # get edges from each face
    
    for facenum in range(len(input_faces)):
        face = input_faces[facenum]
        num_points = len(face)
        # loop over index into face
        for pointindex in range(num_points):
            # if not last point then edge is curr point and next point
            if pointindex < num_points - 1:
                pointnum_1 = face[pointindex]
                pointnum_2 = face[pointindex+1]
            else:
                # for last point edge is curr point and first point
                pointnum_1 = face[pointindex]
                pointnum_2 = face[0]
            # order points in edge by lowest point number
            if pointnum_1 > pointnum_2:
                temp = pointnum_1
                pointnum_1 = pointnum_2
                pointnum_2 = temp
            edges.append([pointnum_1, pointnum_2, facenum])
            
    # sort edges by pointnum_1, pointnum_2, facenum
    
    edges = sorted(edges)
    
    # merge edges with 2 adjacent faces
    # [pointnum_1, pointnum_2, facenum_1, facenum_2] or
    # [pointnum_1, pointnum_2, facenum_1, None]
    
    num_edges = len(edges)
    eindex = 0
    merged_edges = []
    
    while eindex < num_edges:
        e1 = edges[eindex]
        # check if not last edge
        if eindex < num_edges - 1:
            e2 = edges[eindex+1]
            if e1[0] == e2[0] and e1[1] == e2[1]:
                merged_edges.append([e1[0],e1[1],e1[2],e2[2]])
                eindex += 2
            else:
                merged_edges.append([e1[0],e1[1],e1[2],None])
                eindex += 1
        else:
            merged_edges.append([e1[0],e1[1],e1[2],None])
            eindex += 1
            
    # add edge centers
    
    edges_centers = []
    
    for me in merged_edges:
        p1 = input_points[me[0]]
        p2 = input_points[me[1]]
        cp = center_point(p1, p2)
        edges_centers.append(me+[cp])
            
    return edges_centers
       
def get_edge_points(input_points, edges_faces, face_points):
    """
    for each edge, an edge point is created which is the average 
    between the center of the edge and the center of the segment made
    with the face points of the two adjacent faces.
    """
    
    edge_points = []
    
    for edge in edges_faces:
        # get center of edge
        cp = edge[4]
        # get center of two facepoints
        fp1 = face_points[edge[2]]
        # if not two faces just use one facepoint
        # should not happen for solid like a cube
        if edge[3] == None:
            fp2 = fp1
        else:
            fp2 = face_points[edge[3]]
        cfp = center_point(fp1, fp2)
        # get average between center of edge and
        # center of facepoints
        edge_point = center_point(cp, cfp)
        edge_points.append(edge_point)      
        
    return edge_points
    
def get_avg_face_points(input_points, input_faces, face_points):
    """
    
    for each point calculate
    
    the average of the face points of the faces the point belongs to (avg_face_points)
    
    create a list of lists of two numbers [facepoint_sum, num_points] by going through the
    points in all the faces.
    
    then create the avg_face_points list of point by dividing point_sum (x, y, z) by num_points
    
    """
    
    # initialize list with [[0.0, 0.0, 0.0], 0]
    
    num_points = len(input_points)
    
    temp_points = []
    
    for pointnum in range(num_points):
        temp_points.append([[0.0, 0.0, 0.0], 0])
        
    # loop through faces updating temp_points
    
    for facenum in range(len(input_faces)):
        fp = face_points[facenum]
        for pointnum in input_faces[facenum]:
            tp = temp_points[pointnum][0]
            temp_points[pointnum][0] = sum_point(tp,fp)
            temp_points[pointnum][1] += 1
            
    # divide to create avg_face_points
    
    avg_face_points = []
    
    for tp in temp_points:
       afp = div_point(tp[0], tp[1])
       avg_face_points.append(afp)
       
    return avg_face_points
    
def get_avg_mid_edges(input_points, edges_faces):
    """
    
    the average of the centers of edges the point belongs to (avg_mid_edges)
    
    create list with entry for each point 
    each entry has two elements. one is a point that is the sum of the centers of the edges
    and the other is the number of edges. after going through all edges divide by
    number of edges.
    
    """
    
    # initialize list with [[0.0, 0.0, 0.0], 0]
    
    num_points = len(input_points)
    
    temp_points = []
    
    for pointnum in range(num_points):
        temp_points.append([[0.0, 0.0, 0.0], 0])
        
    # go through edges_faces using center updating each point
    
    for edge in edges_faces:
        cp = edge[4]
        for pointnum in [edge[0], edge[1]]:
            tp = temp_points[pointnum][0]
            temp_points[pointnum][0] = sum_point(tp,cp)
            temp_points[pointnum][1] += 1
    
    # divide out number of points to get average
    
    avg_mid_edges = []
        
    for tp in temp_points:
       ame = div_point(tp[0], tp[1])
       avg_mid_edges.append(ame)
       
    return avg_mid_edges

def get_points_faces(input_points, input_faces):
    # initialize list with 0
    
    num_points = len(input_points)
    
    points_faces = []
    
    for pointnum in range(num_points):
        points_faces.append(0)
        
    # loop through faces updating points_faces
    
    for facenum in range(len(input_faces)):
        for pointnum in input_faces[facenum]:
            points_faces[pointnum] += 1
            
    return points_faces

def get_new_points(input_points, points_faces, avg_face_points, avg_mid_edges):
    """
    
    m1 = (n - 3.0) / n
    m2 = 1.0 / n
    m3 = 2.0 / n
    new_coords = (m1 * old_coords)
               + (m2 * avg_face_points)
               + (m3 * avg_mid_edges)
                        
    """
    
    new_points =[]
    
    for pointnum in range(len(input_points)):
        n = points_faces[pointnum]
        m1 = (n - 3.0) / n
        m2 = 1.0 / n
        m3 = 2.0 / n
        old_coords = input_points[pointnum]
        p1 = mul_point(old_coords, m1)
        afp = avg_face_points[pointnum]
        p2 = mul_point(afp, m2)
        ame = avg_mid_edges[pointnum]
        p3 = mul_point(ame, m3)
        p4 = sum_point(p1, p2)
        new_coords = sum_point(p4, p3)
        
        new_points.append(new_coords)
        
    return new_points
    
def switch_nums(point_nums):
    """
    Returns tuple of point numbers
    sorted least to most
    """
    if point_nums[0] < point_nums[1]:
        return point_nums
    else:
        return (point_nums[1], point_nums[0])    

def cmc_subdiv(input_points, input_faces):
    # 1. for each face, a face point is created which is the average of all the points of the face.
    # each entry in the returned list is a point (x, y, z).
    
    face_points = get_face_points(input_points, input_faces)
    
    # get list of edges with 1 or 2 adjacent faces
    # [pointnum_1, pointnum_2, facenum_1, facenum_2, center] or
    # [pointnum_1, pointnum_2, facenum_1, None, center]
    
    edges_faces = get_edges_faces(input_points, input_faces)
    
    # get edge points, a list of points
    
    edge_points = get_edge_points(input_points, edges_faces, face_points)
                    
    # the average of the face points of the faces the point belongs to (avg_face_points)                
                    
    avg_face_points = get_avg_face_points(input_points, input_faces, face_points)
       
    # the average of the centers of edges the point belongs to (avg_mid_edges)
       
    avg_mid_edges = get_avg_mid_edges(input_points, edges_faces) 
       
    # how many faces a point belongs to
    
    points_faces = get_points_faces(input_points, input_faces)
    
    """
    
    m1 = (n - 3) / n
    m2 = 1 / n
    m3 = 2 / n
    new_coords = (m1 * old_coords)
               + (m2 * avg_face_points)
               + (m3 * avg_mid_edges)
                        
    """
        
    new_points = get_new_points(input_points, points_faces, avg_face_points, avg_mid_edges)
        
    """
    
    Then each face is replaced by new faces made with the new points,
    
    for a triangle face (a,b,c):
       (a, edge_point ab, face_point abc, edge_point ca)
       (b, edge_point bc, face_point abc, edge_point ab)
       (c, edge_point ca, face_point abc, edge_point bc)
       
    for a quad face (a,b,c,d):
       (a, edge_point ab, face_point abcd, edge_point da)
       (b, edge_point bc, face_point abcd, edge_point ab)
       (c, edge_point cd, face_point abcd, edge_point bc)
       (d, edge_point da, face_point abcd, edge_point cd)
       
    face_points is a list indexed by face number so that is
    easy to get.
    
    edge_points is a list indexed by the edge number
    which is an index into edges_faces.
    
    need to add face_points and edge points to 
    new_points and get index into each.
    
    then create two new structures
    
    face_point_nums - list indexes by facenum
    whose value is the index into new_points
    
    edge_point num - dictionary with key (pointnum_1, pointnum_2)
    and value is index into new_points
       
    """
    
    # add face points to new_points
    
    face_point_nums = []
    
    # point num after next append to new_points
    next_pointnum = len(new_points)
    
    for face_point in face_points:
        new_points.append(face_point)
        face_point_nums.append(next_pointnum)
        next_pointnum += 1
        
    # add edge points to new_points
    
    edge_point_nums = dict()
    
    for edgenum in range(len(edges_faces)):
        pointnum_1 = edges_faces[edgenum][0]
        pointnum_2 = edges_faces[edgenum][1]
        edge_point = edge_points[edgenum]
        new_points.append(edge_point)
        edge_point_nums[(pointnum_1, pointnum_2)] = next_pointnum
        next_pointnum += 1
    
    # new_points now has the points to output. Need new
    # faces
    
    """
    
    just doing this case for now:
    
    for a quad face (a,b,c,d):
       (a, edge_point ab, face_point abcd, edge_point da)
       (b, edge_point bc, face_point abcd, edge_point ab)
       (c, edge_point cd, face_point abcd, edge_point bc)
       (d, edge_point da, face_point abcd, edge_point cd)
       
    new_faces will be a list of lists where the elements are like this:
    
    [pointnum_1, pointnum_2, pointnum_3, pointnum_4]
    
    """
    
    new_faces =[]
    
    for oldfacenum in range(len(input_faces)):
        oldface = input_faces[oldfacenum]
        # 4 point face
        if len(oldface) == 4:
            a = oldface[0]
            b = oldface[1]
            c = oldface[2]
            d = oldface[3]
            face_point_abcd = face_point_nums[oldfacenum]
            edge_point_ab = edge_point_nums[switch_nums((a, b))]
            edge_point_da = edge_point_nums[switch_nums((d, a))]
            edge_point_bc = edge_point_nums[switch_nums((b, c))]
            edge_point_cd = edge_point_nums[switch_nums((c, d))]
            new_faces.append((a, edge_point_ab, face_point_abcd, edge_point_da))
            new_faces.append((b, edge_point_bc, face_point_abcd, edge_point_ab))
            new_faces.append((c, edge_point_cd, face_point_abcd, edge_point_bc))
            new_faces.append((d, edge_point_da, face_point_abcd, edge_point_cd))    
    
    return new_points, new_faces
    

def graph_output(output_points, output_faces):
    
    fig = plt.figure()
    ax = fig.add_subplot(111, projection='3d')
    
    """
    
    Plot each face
    
    """
    
    for facenum in range(len(output_faces)):
        curr_face = output_faces[facenum]
        xcurr = []
        ycurr = []
        zcurr = []
        for pointnum in range(len(curr_face)):
            xcurr.append(output_points[curr_face[pointnum]][0])
            ycurr.append(output_points[curr_face[pointnum]][1])
            zcurr.append(output_points[curr_face[pointnum]][2])
        xcurr.append(output_points[curr_face[0]][0])
        ycurr.append(output_points[curr_face[0]][1])
        zcurr.append(output_points[curr_face[0]][2])
        
        ax.plot(xcurr,ycurr,zcurr,color='b')
    
    plt.show()


# cube

input_points = [
  [-1.0,  1.0,  1.0],
  [-1.0, -1.0,  1.0],
  [ 1.0, -1.0,  1.0],
  [ 1.0,  1.0,  1.0],
  [ 1.0, -1.0, -1.0],
  [ 1.0,  1.0, -1.0],
  [-1.0, -1.0, -1.0],
  [-1.0,  1.0, -1.0]
]

input_faces = [
  [0, 1, 2, 3],
  [3, 2, 4, 5],
  [5, 4, 6, 7],
  [7, 0, 3, 5],
  [7, 6, 1, 0],
  [6, 1, 2, 4],
]

if len(sys.argv) != 2:
    print("Should have one argument integer number of iterations")
    sys.exit()
else:
    iterations = int(sys.argv[1])
    
    output_points, output_faces = input_points, input_faces
    
    for i in range(iterations):
        output_points, output_faces = cmc_subdiv(output_points, output_faces)
        
graph_output(output_points, output_faces)


  

You may also check:How to resolve the algorithm Anagrams/Deranged anagrams step by step in the Go programming language
You may also check:How to resolve the algorithm Knight's tour step by step in the Tcl programming language
You may also check:How to resolve the algorithm Time a function step by step in the Groovy programming language
You may also check:How to resolve the algorithm Compile-time calculation step by step in the C++ programming language
You may also check:How to resolve the algorithm Character codes step by step in the Babel programming language