Simple data¶
In [58]:
Copied!
import bpy
from IPython.display import display, Image
def fresh_scene(keep_cube=False):
# Deselect all objects
bpy.ops.object.select_all(action='DESELECT')
# Select all objects except cameras and optionally the default cube
for obj in bpy.context.scene.objects:
if obj.type == 'CAMERA':
obj.select_set(False)
elif obj.name == 'Cube' and keep_cube:
obj.select_set(False)
else:
obj.select_set(True)
bpy.ops.object.delete()
# Add light
bpy.ops.object.light_add(type='SUN')
sun = bpy.context.active_object
sun.location = (0, 0, 0)
from math import radians
sun.rotation_euler = (radians(204), radians(-133), radians(-67))
sun.data.energy = 5
def render_result():
bpy.ops.render.render()
bpy.data.images['Render Result'].save_render(filepath="img.png")
display(Image(filename="img.png"))
bpy.context.scene.render.resolution_x = 500
bpy.context.scene.render.resolution_y = 200
import bpy
from IPython.display import display, Image
def fresh_scene(keep_cube=False):
# Deselect all objects
bpy.ops.object.select_all(action='DESELECT')
# Select all objects except cameras and optionally the default cube
for obj in bpy.context.scene.objects:
if obj.type == 'CAMERA':
obj.select_set(False)
elif obj.name == 'Cube' and keep_cube:
obj.select_set(False)
else:
obj.select_set(True)
bpy.ops.object.delete()
# Add light
bpy.ops.object.light_add(type='SUN')
sun = bpy.context.active_object
sun.location = (0, 0, 0)
from math import radians
sun.rotation_euler = (radians(204), radians(-133), radians(-67))
sun.data.energy = 5
def render_result():
bpy.ops.render.render()
bpy.data.images['Render Result'].save_render(filepath="img.png")
display(Image(filename="img.png"))
bpy.context.scene.render.resolution_x = 500
bpy.context.scene.render.resolution_y = 200
In [59]:
Copied!
import bpy
import subprocess
import sys
# Path to Blender's Python executable
python_executable = sys.executable
try:
subprocess.check_call([python_executable, "-m", "pip", "install", "polars"])
subprocess.check_call([python_executable, "-m", "pip", "install", "matplotlib"])
subprocess.check_call([python_executable, "-m", "pip", "install", "databpy"])
print("Done")
except Exception as e:
print(f"An error occurred: {e}")
import bpy
import subprocess
import sys
# Path to Blender's Python executable
python_executable = sys.executable
try:
subprocess.check_call([python_executable, "-m", "pip", "install", "polars"])
subprocess.check_call([python_executable, "-m", "pip", "install", "matplotlib"])
subprocess.check_call([python_executable, "-m", "pip", "install", "databpy"])
print("Done")
except Exception as e:
print(f"An error occurred: {e}")
Requirement already satisfied: polars in /Applications/Blender.app/Contents/Resources/4.3/python/lib/python3.11/site-packages (1.19.0)
[notice] A new release of pip is available: 24.0 -> 24.3.1 [notice] To update, run: /Applications/Blender.app/Contents/Resources/4.3/python/bin/python3.11 -m pip install --upgrade pip
Requirement already satisfied: matplotlib in /Applications/Blender.app/Contents/Resources/4.3/python/lib/python3.11/site-packages (3.10.0) Requirement already satisfied: contourpy>=1.0.1 in /Applications/Blender.app/Contents/Resources/4.3/python/lib/python3.11/site-packages (from matplotlib) (1.3.1) Requirement already satisfied: cycler>=0.10 in /Applications/Blender.app/Contents/Resources/4.3/python/lib/python3.11/site-packages (from matplotlib) (0.12.1) Requirement already satisfied: fonttools>=4.22.0 in /Applications/Blender.app/Contents/Resources/4.3/python/lib/python3.11/site-packages (from matplotlib) (4.55.3) Requirement already satisfied: kiwisolver>=1.3.1 in /Applications/Blender.app/Contents/Resources/4.3/python/lib/python3.11/site-packages (from matplotlib) (1.4.8) Requirement already satisfied: numpy>=1.23 in /Applications/Blender.app/Contents/Resources/4.3/python/lib/python3.11/site-packages (from matplotlib) (1.24.3) Requirement already satisfied: packaging>=20.0 in /Users/jan-hendrik/.local/lib/python3.11/site-packages (from matplotlib) (24.1) Requirement already satisfied: pillow>=8 in /Applications/Blender.app/Contents/Resources/4.3/python/lib/python3.11/site-packages (from matplotlib) (11.1.0) Requirement already satisfied: pyparsing>=2.3.1 in /Applications/Blender.app/Contents/Resources/4.3/python/lib/python3.11/site-packages (from matplotlib) (3.2.1) Requirement already satisfied: python-dateutil>=2.7 in /Users/jan-hendrik/.local/lib/python3.11/site-packages (from matplotlib) (2.9.0.post0) Requirement already satisfied: six>=1.5 in /Applications/Blender.app/Contents/Resources/4.3/python/lib/python3.11/site-packages (from python-dateutil>=2.7->matplotlib) (1.17.0)
[notice] A new release of pip is available: 24.0 -> 24.3.1 [notice] To update, run: /Applications/Blender.app/Contents/Resources/4.3/python/bin/python3.11 -m pip install --upgrade pip
Requirement already satisfied: databpy in /Applications/Blender.app/Contents/Resources/4.3/python/lib/python3.11/site-packages (0.0.8) Requirement already satisfied: numpy<2.0,>=1.24.0 in /Applications/Blender.app/Contents/Resources/4.3/python/lib/python3.11/site-packages (from databpy) (1.24.3) Done
[notice] A new release of pip is available: 24.0 -> 24.3.1 [notice] To update, run: /Applications/Blender.app/Contents/Resources/4.3/python/bin/python3.11 -m pip install --upgrade pip
In [60]:
Copied!
import polars as pl
url = 'https://raw.githubusercontent.com/kolibril13/ipydrop/refs/heads/main/dino.csv'
df1 = pl.read_csv(url)
length_data = len(df1)
print(f"Loaded {length_data} rows")
df1.head()
import polars as pl
url = 'https://raw.githubusercontent.com/kolibril13/ipydrop/refs/heads/main/dino.csv'
df1 = pl.read_csv(url)
length_data = len(df1)
print(f"Loaded {length_data} rows")
df1.head()
Loaded 142 rows
Out[60]:
shape: (5, 2)
x | y |
---|---|
f64 | f64 |
55.3846 | 97.1795 |
51.5385 | 96.0256 |
46.1538 | 94.4872 |
42.8205 | 91.4103 |
40.7692 | 88.3333 |
In [61]:
Copied!
import matplotlib.pyplot as plt
plt.style.use('_mpl-gallery')
import matplotlib.pyplot as plt
plt.style.use('_mpl-gallery')
In [62]:
Copied!
# Extract x and y from the DataFrame
x_values1 = df1['x'] / 10
y_values1 = df1['y'] / 10 * 0.8
# Create the scatter plot using plt.scatter
plt.axis('equal');
plt.scatter(x_values1, y_values1);
# Extract x and y from the DataFrame
x_values1 = df1['x'] / 10
y_values1 = df1['y'] / 10 * 0.8
# Create the scatter plot using plt.scatter
plt.axis('equal');
plt.scatter(x_values1, y_values1);
In [63]:
Copied!
def camera_from_above():
camera = bpy.context.scene.camera
camera.location = (6, 4, 10) # Position above the XY plane
camera.rotation_euler = (0, 0, 0) # Rotate to look at XY plane
# Set the camera to orthographic mode
camera.data.type = 'ORTHO'
# Set the orthographic scale to 12
camera.data.ortho_scale = 9
bpy.context.scene.render.resolution_x = 300
bpy.context.scene.render.resolution_y = 300
def camera_from_above():
camera = bpy.context.scene.camera
camera.location = (6, 4, 10) # Position above the XY plane
camera.rotation_euler = (0, 0, 0) # Rotate to look at XY plane
# Set the camera to orthographic mode
camera.data.type = 'ORTHO'
# Set the orthographic scale to 12
camera.data.ortho_scale = 9
bpy.context.scene.render.resolution_x = 300
bpy.context.scene.render.resolution_y = 300
In [64]:
Copied!
fresh_scene()
for (x, y) in zip(x_values1,y_values1):
bpy.ops.mesh.primitive_uv_sphere_add(radius=0.2, location=(x, y, 0))
camera_from_above()
render_result()
fresh_scene()
for (x, y) in zip(x_values1,y_values1):
bpy.ops.mesh.primitive_uv_sphere_add(radius=0.2, location=(x, y, 0))
camera_from_above()
render_result()
In [65]:
Copied!
fresh_scene()
mat = bpy.data.materials.new(name="GreenMaterial")
mat.diffuse_color = (0, 0, 1, 1) # Green color (R, G, B, A)
for (x, y) in zip(x_values1,y_values1):
bpy.ops.mesh.primitive_uv_sphere_add(radius=0.2, location=(x, y, 0))
bpy.context.active_object.data.materials.append(mat)
bpy.context.scene.render.resolution_x = 300
bpy.context.scene.render.resolution_y = 300
# Add a white plane behind the spheres
bpy.ops.mesh.primitive_plane_add(size=20, location=(5, 5, 0))
plane = bpy.context.active_object
# Create a white material and assign it to the plane
white_mat = bpy.data.materials.new(name="WhiteMaterial")
white_mat.diffuse_color = (1, 1, 1, 1) # White color (R, G, B, A)
plane.data.materials.append(white_mat)
render_result()
fresh_scene()
mat = bpy.data.materials.new(name="GreenMaterial")
mat.diffuse_color = (0, 0, 1, 1) # Green color (R, G, B, A)
for (x, y) in zip(x_values1,y_values1):
bpy.ops.mesh.primitive_uv_sphere_add(radius=0.2, location=(x, y, 0))
bpy.context.active_object.data.materials.append(mat)
bpy.context.scene.render.resolution_x = 300
bpy.context.scene.render.resolution_y = 300
# Add a white plane behind the spheres
bpy.ops.mesh.primitive_plane_add(size=20, location=(5, 5, 0))
plane = bpy.context.active_object
# Create a white material and assign it to the plane
white_mat = bpy.data.materials.new(name="WhiteMaterial")
white_mat.diffuse_color = (1, 1, 1, 1) # White color (R, G, B, A)
plane.data.materials.append(white_mat)
render_result()
this works, but it's very limited! E.g.
- not possible to scale individual spheres.
- not possible to use geometry nodes with this object.
Can we maybe add this geometry nodes setup?
Loading Pre-Defined Geometry Nodes¶
In [66]:
Copied!
# option 1: Load the node group to the current blend scene
#TODO Make this it's own utility tool
import bpy
# Path to the blend file
blend_file_path = "docs/cube_gn_position.blend"
node_group_name = "gn_place_spheres"
# Load only the node group from the blend file
with bpy.data.libraries.load(blend_file_path, link=False) as (data_from, data_to):
if node_group_name in data_from.node_groups:
data_to.node_groups = [node_group_name]
# Access the node group
node_group_place_spheres = bpy.data.node_groups[node_group_name]
# Print all the node names inside the node group
print(f"Nodes in node group '{node_group_name}':")
for node in node_group_place_spheres.nodes:
print(node.name)
# option 1: Load the node group to the current blend scene
#TODO Make this it's own utility tool
import bpy
# Path to the blend file
blend_file_path = "docs/cube_gn_position.blend"
node_group_name = "gn_place_spheres"
# Load only the node group from the blend file
with bpy.data.libraries.load(blend_file_path, link=False) as (data_from, data_to):
if node_group_name in data_from.node_groups:
data_to.node_groups = [node_group_name]
# Access the node group
node_group_place_spheres = bpy.data.node_groups[node_group_name]
# Print all the node names inside the node group
print(f"Nodes in node group '{node_group_name}':")
for node in node_group_place_spheres.nodes:
print(node.name)
Nodes in node group 'gn_place_spheres': Group Input Group Output Instance on Points UV Sphere
In [67]:
Copied!
# option 2: Open the compeletely new blend file and access the node group
#bpy.ops.wm.open_mainfile(filepath="cube_gn_position.blend")
# Access the node group
#node_group_place_spheres = bpy.data.node_groups["gn_place_spheres"]
# Print all the node names inside the node group
#print("Nodes in node group 'gn_place_spheres':")
#for node in node_group_place_spheres.nodes:
# print(node.name)
# option 2: Open the compeletely new blend file and access the node group
#bpy.ops.wm.open_mainfile(filepath="cube_gn_position.blend")
# Access the node group
#node_group_place_spheres = bpy.data.node_groups["gn_place_spheres"]
# Print all the node names inside the node group
#print("Nodes in node group 'gn_place_spheres':")
#for node in node_group_place_spheres.nodes:
# print(node.name)
In [68]:
Copied!
fresh_scene()
import mathutils
import databpy as db
# Old way
# Create a new mesh and object for the point cloud
#mesh = bpy.data.meshes.new("HelloMesh")
#my_point_obj = bpy.data.objects.new("HelloPoint", mesh)
#points = [mathutils.Vector((6, 4, 0))] # One point at coordinates
# Apply the points to the mesh
#mesh.from_pydata(vertices = points, edges=[], faces=[])
#mesh.update()
#bpy.context.collection.objects.link(my_point_obj)
#new way:
import numpy as np
import databpy as db
single_vertex = np.array([[6, 4, 0]])
# Create a mesh object with the single vertex
my_point_obj = db.create_object(single_vertex, name="SinglePoint")
# Print the name of the created object
print(my_point_obj.name)
bpy.context.scene.render.resolution_x = 300
bpy.context.scene.render.resolution_y = 200
render_result()
fresh_scene()
import mathutils
import databpy as db
# Old way
# Create a new mesh and object for the point cloud
#mesh = bpy.data.meshes.new("HelloMesh")
#my_point_obj = bpy.data.objects.new("HelloPoint", mesh)
#points = [mathutils.Vector((6, 4, 0))] # One point at coordinates
# Apply the points to the mesh
#mesh.from_pydata(vertices = points, edges=[], faces=[])
#mesh.update()
#bpy.context.collection.objects.link(my_point_obj)
#new way:
import numpy as np
import databpy as db
single_vertex = np.array([[6, 4, 0]])
# Create a mesh object with the single vertex
my_point_obj = db.create_object(single_vertex, name="SinglePoint")
# Print the name of the created object
print(my_point_obj.name)
bpy.context.scene.render.resolution_x = 300
bpy.context.scene.render.resolution_y = 200
render_result()
SinglePoint
In [69]:
Copied!
modifier = my_point_obj.modifiers.new(name="GeometryNodes", type='NODES')
modifier.node_group = node_group_place_spheres
render_result()
modifier = my_point_obj.modifiers.new(name="GeometryNodes", type='NODES')
modifier.node_group = node_group_place_spheres
render_result()
In [70]:
Copied!
my_point_obj.modifiers["GeometryNodes"]["Socket_2"] = 4 # set radius
render_result()
my_point_obj.modifiers["GeometryNodes"]["Socket_2"] = 4 # set radius
render_result()
In [71]:
Copied!
fresh_scene()
single_vertex = np.array([(x, y, 0) for x, y in zip(x_values1, y_values1)])
my_point_obj = db.create_object(single_vertex, name="Deno")
mesh = bpy.data.meshes.new("HelloDenoMesh")
my_point_obj = bpy.data.objects.new("HelloDeno", mesh)
points = [mathutils.Vector((x, y, 0)) for x, y in zip(x_values1, y_values1)]
# Apply the points to the mesh
mesh.from_pydata(vertices=points, edges=[], faces=[])
mesh.update()
bpy.context.collection.objects.link(my_point_obj)
modifier = my_point_obj.modifiers.new(name="GeometryNodes", type='NODES')
modifier.node_group = node_group_place_spheres
my_point_obj.modifiers["GeometryNodes"]["Socket_2"] = 0.2 # Set radius
camera_from_above()
render_result()
fresh_scene()
single_vertex = np.array([(x, y, 0) for x, y in zip(x_values1, y_values1)])
my_point_obj = db.create_object(single_vertex, name="Deno")
mesh = bpy.data.meshes.new("HelloDenoMesh")
my_point_obj = bpy.data.objects.new("HelloDeno", mesh)
points = [mathutils.Vector((x, y, 0)) for x, y in zip(x_values1, y_values1)]
# Apply the points to the mesh
mesh.from_pydata(vertices=points, edges=[], faces=[])
mesh.update()
bpy.context.collection.objects.link(my_point_obj)
modifier = my_point_obj.modifiers.new(name="GeometryNodes", type='NODES')
modifier.node_group = node_group_place_spheres
my_point_obj.modifiers["GeometryNodes"]["Socket_2"] = 0.2 # Set radius
camera_from_above()
render_result()
In [72]:
Copied!
my_point_obj.modifiers["GeometryNodes"]["Socket_2"] = 0.1 # Set radius
render_result()
my_point_obj.modifiers["GeometryNodes"]["Socket_2"] = 0.1 # Set radius
render_result()
In [73]:
Copied!
# 💡 Now we can squeeze or change anything in geonodes!
render_result()
# 💡 Now we can squeeze or change anything in geonodes!
render_result()
In [74]:
Copied!
# Also possible to change the data points
import pandas as pd
url = 'https://raw.githubusercontent.com/kolibril13/ipydrop/refs/heads/main/star.csv'
df = pd.read_csv(url)
df.head()
# Also possible to change the data points
import pandas as pd
url = 'https://raw.githubusercontent.com/kolibril13/ipydrop/refs/heads/main/star.csv'
df = pd.read_csv(url)
df.head()
Out[74]:
x | y | |
---|---|---|
0 | 58.213608 | 91.881892 |
1 | 58.196054 | 92.214989 |
2 | 58.718231 | 90.310532 |
3 | 57.278373 | 89.907607 |
4 | 58.082020 | 92.008145 |
In [75]:
Copied!
x_values = df['x'] / 10
y_values = df['y'] / 10 * 0.8
plt.axis('equal');
plt.scatter(x_values, y_values);
x_values = df['x'] / 10
y_values = df['y'] / 10 * 0.8
plt.axis('equal');
plt.scatter(x_values, y_values);
In [76]:
Copied!
# Only update the vertex positions without clearing the geometry
for i, (x, y) in enumerate(zip(x_values, y_values)):
my_point_obj.data.vertices[i].co = (x, y, 0)
# Update the mesh and force Blender to recalculate
my_point_obj.data.update()
bpy.context.view_layer.update()
# Only update the vertex positions without clearing the geometry
for i, (x, y) in enumerate(zip(x_values, y_values)):
my_point_obj.data.vertices[i].co = (x, y, 0)
# Update the mesh and force Blender to recalculate
my_point_obj.data.update()
bpy.context.view_layer.update()
In [77]:
Copied!
render_result()
render_result()
[I 2025-01-13 20:32:27.399 ServerApp] Saving file at /docs/n2data_simple.ipynb
In [ ]:
Copied!