Generate wall library.py: Difference between revisions
Jump to navigation
Jump to search
No edit summary |
No edit summary |
||
| Line 3: | Line 3: | ||
from pathlib import Path | from pathlib import Path | ||
import sys | import sys | ||
import yaml | import yaml | ||
import FreeCAD | import FreeCAD | ||
import Part | import Part | ||
| Line 12: | Line 10: | ||
IN = 25.4 | IN = 25.4 | ||
| Line 21: | Line 18: | ||
def nominal_to_actual_lumber(nominal): | def nominal_to_actual_lumber(nominal): | ||
table = { | table = { | ||
"2x2": (1.5, 1.5), | "2x2": (1.5, 1.5), | ||
| Line 30: | Line 28: | ||
"2x12": (1.5, 11.25), | "2x12": (1.5, 11.25), | ||
} | } | ||
return table[nominal] | return table[nominal] | ||
def in_to_mm( | def in_to_mm(v): | ||
return | return v * IN | ||
def | def ft_to_in(v): | ||
return | return v * 12 | ||
def make_box(x_len, y_len, z_len, x, y, z): | def make_box(x_len, y_len, z_len, x, y, z): | ||
shape = Part.makeBox(x_len, y_len, z_len) | shape = Part.makeBox(x_len, y_len, z_len) | ||
shape.translate(FreeCAD.Vector(x, y, z)) | shape.translate(FreeCAD.Vector(x, y, z)) | ||
return shape | return shape | ||
| Line 77: | Line 78: | ||
stud_thickness_in, stud_depth_in = nominal_to_actual_lumber(stud_nominal) | stud_thickness_in, stud_depth_in = nominal_to_actual_lumber(stud_nominal) | ||
width_in = width_ft | width_in = ft_to_in(width_ft) | ||
framed_height_in = height_ft | framed_height_in = ft_to_in(height_ft) | ||
plate_thickness_in = stud_thickness_in | plate_thickness_in = stud_thickness_in | ||
stud_length_in = framed_height_in - 2 * plate_thickness_in | stud_length_in = framed_height_in - (2 * plate_thickness_in) | ||
doc = FreeCAD.newDocument(instance_id) | doc = FreeCAD.newDocument(instance_id) | ||
| Line 94: | Line 95: | ||
framed_height_mm = in_to_mm(framed_height_in) | framed_height_mm = in_to_mm(framed_height_in) | ||
# Bottom plate | |||
bottom_plate = make_box( | bottom_plate = make_box( | ||
width_mm, | width_mm, | ||
| Line 102: | Line 104: | ||
0 | 0 | ||
) | ) | ||
shapes.append(("bottom_plate", bottom_plate)) | shapes.append(("bottom_plate", bottom_plate)) | ||
# Top plate | |||
top_plate = make_box( | top_plate = make_box( | ||
width_mm, | width_mm, | ||
| Line 112: | Line 116: | ||
framed_height_mm - plate_thickness_mm | framed_height_mm - plate_thickness_mm | ||
) | ) | ||
shapes.append(("top_plate", top_plate)) | shapes.append(("top_plate", top_plate)) | ||
# Studs | |||
stud_positions = derive_stud_positions(width_in, stud_thickness_in, spacing_oc_in) | stud_positions = derive_stud_positions(width_in, stud_thickness_in, spacing_oc_in) | ||
| Line 129: | Line 135: | ||
shapes.append((f"stud_{i+1:02d}", stud)) | shapes.append((f"stud_{i+1:02d}", stud)) | ||
# OSB | |||
osb = make_box( | osb = make_box( | ||
width_mm, | width_mm, | ||
| Line 140: | Line 147: | ||
shapes.append(("osb_exterior", osb)) | shapes.append(("osb_exterior", osb)) | ||
# Add objects to document | |||
for name, shape in shapes: | for name, shape in shapes: | ||
obj = doc.addObject("Part::Feature", name) | obj = doc.addObject("Part::Feature", name) | ||
obj.Shape = shape | obj.Shape = shape | ||
# ensure objects are visible when opened in FreeCAD | |||
if hasattr(obj, "ViewObject"): | |||
obj.ViewObject.Visibility = True | |||
doc.recompute() | doc.recompute() | ||
| Line 150: | Line 163: | ||
def save_document(doc, path): | def save_document(doc, path): | ||
path.parent.mkdir(exist_ok=True) | path.parent.mkdir(exist_ok=True) | ||
doc.saveAs(str(path)) | doc.saveAs(str(path)) | ||
Revision as of 02:44, 11 March 2026
- !/usr/bin/env python3
from pathlib import Path import sys import yaml import FreeCAD import Part
OUTPUT_DIR = Path("cad_library")
IN = 25.4
def load_yaml(path):
with open(path, "r") as f:
return yaml.safe_load(f)
def nominal_to_actual_lumber(nominal):
table = {
"2x2": (1.5, 1.5),
"2x3": (1.5, 2.5),
"2x4": (1.5, 3.5),
"2x6": (1.5, 5.5),
"2x8": (1.5, 7.25),
"2x10": (1.5, 9.25),
"2x12": (1.5, 11.25),
}
return table[nominal]
def in_to_mm(v):
return v * IN
def ft_to_in(v):
return v * 12
def make_box(x_len, y_len, z_len, x, y, z):
shape = Part.makeBox(x_len, y_len, z_len) shape.translate(FreeCAD.Vector(x, y, z))
return shape
def derive_stud_positions(width_in, stud_thickness_in, spacing_oc_in):
positions = [0.0]
right_edge = width_in - stud_thickness_in current = spacing_oc_in
while current + stud_thickness_in <= right_edge:
positions.append(current)
current += spacing_oc_in
if positions[-1] != right_edge:
positions.append(right_edge)
return positions
def build_wall(instance):
instance_id = instance["id"] params = instance["parameters"]
width_ft = params["nominal_width_ft"] height_ft = params["nominal_height_ft"] stud_nominal = params["stud_lumber_nominal"] spacing_oc_in = params["stud_spacing_oc_in"] osb_thickness_in = params["osb_thickness_in"]
stud_thickness_in, stud_depth_in = nominal_to_actual_lumber(stud_nominal)
width_in = ft_to_in(width_ft) framed_height_in = ft_to_in(height_ft)
plate_thickness_in = stud_thickness_in stud_length_in = framed_height_in - (2 * plate_thickness_in)
doc = FreeCAD.newDocument(instance_id)
shapes = []
width_mm = in_to_mm(width_in) stud_depth_mm = in_to_mm(stud_depth_in) plate_thickness_mm = in_to_mm(plate_thickness_in) stud_length_mm = in_to_mm(stud_length_in) osb_thickness_mm = in_to_mm(osb_thickness_in) framed_height_mm = in_to_mm(framed_height_in)
# Bottom plate
bottom_plate = make_box(
width_mm,
stud_depth_mm,
plate_thickness_mm,
0,
0,
0
)
shapes.append(("bottom_plate", bottom_plate))
# Top plate
top_plate = make_box(
width_mm,
stud_depth_mm,
plate_thickness_mm,
0,
0,
framed_height_mm - plate_thickness_mm
)
shapes.append(("top_plate", top_plate))
# Studs stud_positions = derive_stud_positions(width_in, stud_thickness_in, spacing_oc_in)
for i, x in enumerate(stud_positions):
stud = make_box(
in_to_mm(stud_thickness_in),
stud_depth_mm,
stud_length_mm,
in_to_mm(x),
0,
plate_thickness_mm
)
shapes.append((f"stud_{i+1:02d}", stud))
# OSB
osb = make_box(
width_mm,
osb_thickness_mm,
framed_height_mm,
0,
stud_depth_mm,
0
)
shapes.append(("osb_exterior", osb))
# Add objects to document for name, shape in shapes:
obj = doc.addObject("Part::Feature", name)
obj.Shape = shape
# ensure objects are visible when opened in FreeCAD
if hasattr(obj, "ViewObject"):
obj.ViewObject.Visibility = True
doc.recompute()
return doc
def save_document(doc, path):
path.parent.mkdir(exist_ok=True) doc.saveAs(str(path))
def main():
if len(sys.argv) != 2:
print("Usage: generate_wall_library.py instances.yaml")
sys.exit(1)
instances_doc = load_yaml(sys.argv[1]) instances = instances_doc["instances"]
OUTPUT_DIR.mkdir(exist_ok=True)
for inst in instances:
instance_id = inst["id"]
print(f"Generating {instance_id}...")
doc = build_wall(inst)
out_path = OUTPUT_DIR / f"{instance_id}.FCStd"
save_document(doc, out_path)
print(f"Saved {out_path}")
print(f"Generated {len(instances)} modules")
if __name__ == "__main__":
main()