OpenFlexure Microscope OpenSCAD docs

libs/compact_nut_seat.scad

An attempt at an alternative to my ageing "nut_seat_with_flex" design...

(c) 2016 Richard Bowman - released under CERN Open Hardware License

function actuator_nut_size()
[Source]
function actuator_nut_size() = 3;

* * Nominal thread size of the actuator nut

function actuator_shaft_radius()
[Source]
function actuator_shaft_radius() = actuator_nut_size()/2 * 1.15;

* * Radius of hole to cut for actuator screw

function column_base_radius()
[Source]
function column_base_radius() = actuator_shaft_radius() + 2;

* * Radius of the bottom of the actuator column * Note that this sets width of the actuator column. However, the column * itself is stretched into an oval to match the depth of the top cube of * the column (i.e. where the nut trap is)

function actuator_dims(params)
[Source]
function actuator_dims(params) = let(
    width = column_base_radius()*2
 ) [width, actuating_nut_r(params), 6];

* * The dimensions of the actuating lever for the x and y axes

function actuator_nut_slot_size()
[Source]
function actuator_nut_slot_size() = let(
    //maximum width of the m3 nut nut (flat to flat) specified by ISO 4032
    nut_w = 5.5,
    // Multiplying by a clearance factor of 1.091 that has been tested empirically
    // for many years, to provide good grip on the nuts when in the top of the
    // trap which is 90% of this slot size.
    nut_w_clear = nut_w*1.091,
    // ISO 4032 specifies a maximum height for the m3 nut of of 2.4mm
    nut_h = 2.4
) [nut_w_clear, nut_w_clear/sin(60), nut_h+0.6];

* * The dimensions of the nut slot in the actuator * Returns a vector giving the space for the nut including clearnce. * [space across flats, space across corners, space for height]

function column_core_size()
[Source]
function column_core_size() = let(
    nut_slot_xy =  zero_z(actuator_nut_slot_size()),
    // Adding extra material to the column. Note, must leave z=0 here
    // 1.5 is added in x and y as the top of the actuator column is 3mm wider
    // than the nut slot. The 7 relates to the size of the hook, and the other
    // number set the clearance.
    extra_xy =  2*[1.5+7+1, 1.5+1.5, 0]
) nut_slot_xy + extra_xy;
function actuator_wall_t()
[Source]
function actuator_wall_t() = 1.6;

* * Thickness of the actuator housing wall

function actuator_housing_xy_size()
[Source]
function actuator_housing_xy_size() = let(
    extra_size_for_walls = [2, 2, 0]*actuator_wall_t()
) column_core_size() + extra_size_for_walls;

* * The xy size of the the actuator housing (outer dimensions) * Note that this returns a 3-vector with z=0

function actuator_entry_width()
[Source]
function actuator_entry_width() = 2*column_base_radius()+3;
module nut_trap_and_slot(
    r,
    slot,
    squeeze=0.9,
    trap_h=undef,
    slot_length=999,
    include_bridged_top=true,
    bottom_hole=0
)
[Source]
module nut_trap_and_slot(r, slot, squeeze=0.9, trap_h=undef, slot_length=999, include_bridged_top=true, bottom_hole=0){
    hole_r = r*1.15/2;
    trap_height = if_undefined_set_default(trap_h, r);
    w = slot.x; //width of the nut entry slot (should be slightly larger than the nut)
    l = slot.y; //length/depth of the slot (now ignored)
    h = slot.z; //height of the slot
    r1 = w/2/cos(30); //bottom of nut trap is large
    r2 = r*squeeze; //top of nut trap is very tight
    sequential_hull(){
        if (slot_length>0){
            translate([-w/2, slot_length, 0]){
                cube([w,tiny(),h]);
            }
        }
        union(){
            if (slot_length>0){
                translate([-w/2,l/2-tiny(),0]){
                    cube([w,tiny(),h]);
                }
            }
            rotate(30){
                translate_z(-bottom_hole){
                    cylinder(d=w/sin(60), h=h+bottom_hole, $fn=6);
                }
            }
        }
        a = 1/trap_height;
        rotate(30){
            cylinder(r=r1*(1-a) + r2*a, h=h+1, $fn=6);
        }
        rotate(30){
            cylinder(r=r2, h=h+trap_height, $fn=6);
        }
    }
    // ensure the hole in the top can be made nicely
    if (include_bridged_top) {
        intersection(){
            translate([-2*r2, -hole_r,0]){
                cube([4*r2, 2*hole_r, h + trap_height + 0.5]);
            }
            rotate(30){
                cylinder(r=r2, h= h+trap_height + 1, $fn=6);
            }
        }
    }
}
module m3_nut_trap_with_shaft(
    slot_angle=0,
    tilt=0,
    deep_shaft=0,
    slot_length=999,
    chamfer_offset=undef
)
[Source]
module m3_nut_trap_with_shaft(slot_angle=0,tilt=0,deep_shaft=0,slot_length=999,chamfer_offset=undef)
{
    // Nut trap for an M3 nut with a screw from the top this is a
    // solid object difference it from your part.
    // Trap starts at z=1mm and ends at 7.5mm
    // We recommend have the outer stucture occupies the space from z = 0-9mm
    //
    // For instances where the slot intersects a curved outer surface, a chamfer 
    // option is provided to reduce the occurance of curved unsupported bridges

    rotate_x(tilt){
        rotate_z(slot_angle){
            translate_z(1){
                union(){
                    nut_trap_and_slot(actuator_nut_size(), actuator_nut_slot_size(), slot_length=slot_length);
                    translate_z(-deep_shaft){
                        cylinder(r=actuator_shaft_radius(), h=99, $fn=16);
                    }
                    // chamfer up
                    if (!is_undef(chamfer_offset)){
                        x_size = actuator_nut_slot_size().x;
                        z_size = actuator_nut_size();
                        y_offset = chamfer_offset + 1; // makes the offset from approx hole centre
                        angle = 30;
                        translate([-x_size/2, y_offset, 0.1]){
                            hull(){
                                rotate_x(angle){
                                    cube([x_size, 9, z_size]);
                                }
                                cube([x_size, 9, z_size]);
                            }
                        }
                    }
                }
            }
        }
    }
}
module central_actuator_column(h, top)
[Source]
module central_actuator_column(h, top){
    //The central column of the actuator including the square head. The column extends down
    //past the bottom of the base and must be cut
    $fn=16;
    r1 = column_base_radius(); //size of the bottom part
    r2 = sqrt(top.x*top.x+top.y*top.y)/2; //outer radius of top
    sequential_hull(){
        translate_z(-99){
            resize([2*r1, top.y, tiny()]){
                cylinder(r=r1, h=tiny());
            }
        }
        translate_z(h-top.z - 2*(r2-r1)){
            resize([2*r1, top.y, tiny()]){
                cylinder(r=r1, h=tiny());
            }
        }
        translate_z(h-top.z/2){
            cube(top, center=true);
        }
    }
}
module actuator_hooks(h, top)
[Source]
module actuator_hooks(h,top){
    //These are the hooks on the actuator
    //Reflect to get two hooks
    reflect_x(){
        //Translate to the correct postion on the actuator
        translate([top.x/2,0,h]){
            //Mirror as build upside down
            mirror([0,0,1]){
                // The hook is the sequantiall hull of:
                sequential_hull(){
                    //A thin cube on the side wall of the block
                    translate([-tiny(),-top.y/2,0]){
                        cube([tiny(),top.y,top.z]);
                    }
                    //A thin cylinder inside the block so the nex section is thin
                    translate_z(0.5){
                        scale([0.5 ,1, 1]){
                            cylinder(d=4.5, h=top.z-2);
                        }
                    }
                    //A compressed truncated cone just outside the block
                    translate([1.5,0,0.5]){
                        resize([3,4,3.5]){
                            cylinder(d1=1, d2=4, h=4);
                        }
                    }
                    //Another compressed truncated cone just under where the
                    //hook rises
                    translate([3.5,0,0.5]){
                        resize([2.5,3.0,1.5]){
                            cylinder(d1=1,d2=3.5);
                        }
                    }
                    // A tri-lobular shape for the top of the hook formed from the union
                    // of three cylinders.
                    union(){
                        reflect_y(){
                            translate([4.5,0.5,0]){
                                cylinder(d=1,h=1);
                            }
                        }
                        translate_x(4){
                            cylinder(d=1,h=1);
                        }
                    }
                }
            }
        }
    }
}
module actuator_ties(tilt=0, lever_tip=3)
[Source]
module actuator_ties(tilt=0, lever_tip=3){
    // The ties for the actuator.
    rotate_x(tilt){
        translate_z(lever_tip+flex_dims().z+3){
            cube([actuator_housing_xy_size().x-actuator_wall_t(), 1, 0.5], center=true);
        }
    }
}
module actuator_column(
    h,
    tilt=0,
    lever_tip=3,
    flip_nut_slot=false,
    join_to_casing=true,
    no_voids=false
)
[Source]
module actuator_column(h, tilt=0, lever_tip=3, flip_nut_slot=false, join_to_casing=true, no_voids=false){
    // An "actuator column", a nearly-vertical tower, with a nut trap and hooks
    // for elastic bands at the top, usually attached to a flexure at the bottom.
    // There's often one of these inside the casing under an adjustment screw/gear
    //h: the height of the column
    //tilt: the column is rotated about the x axis
    //lever_tip: height of the actuating lever at its end (can taper up at 45 degrees)
    //flip_nut_slot: if set to true, the nut is inserted from -y
    //join_to_casing: if set to true, the column is joined to the casing by thin threads
    //no_voids: don't leave a void for the nut or screw, used for the drilling jig.

    top = actuator_nut_slot_size() + [3,3,actuator_nut_size() + 1.5]; //size of the top part
    slot_angle = flip_nut_slot ? 180 : 0; //enter from -y if needed
    $fn=16;
    difference(){
        union(){
            rotate_x(tilt){
                central_actuator_column(h, top);
                // hooks for elastic bands/springs
                actuator_hooks(h, top);
            }
            // join the column to the casing, for strength during printing
            // This module does the tilt itself so it can be rendered separately
            // for instructions
            if(join_to_casing){
                actuator_ties(tilt, lever_tip);
            }
        }

        // nut trap
        if(!no_voids){
            rotate_x(tilt){
                rotate(slot_angle){
                    translate_z(h-top.z){
                        nut_trap_and_slot(actuator_nut_size(), actuator_nut_slot_size());
                    }
                }
            }
        }

        // shaft for the screw
        // NB this is raised up from the bottom so it stays within the shaft - this may need to change depending on the length of screw we use...
        if(!no_voids){
            rotate_x(tilt){
                translate_z(lever_tip){
                    cylinder(r=actuator_shaft_radius(), h=999);
                    translate_z(-lever_tip+1){
                        //pointy bottom (stronger)
                        cylinder(r1=0, r2=actuator_shaft_radius(), h=lever_tip-1);
                    }
                }
            }
        }

        // space for lever and flexure
        translate([-99, -flex_dims().y/2, flex_dims().z]){
            sequential_hull(){
                cube([999,flex_dims().y,lever_tip]);
                translate([0,-999,999]){
                    cube([999,flex_dims().y,lever_tip]);
                }
            }
        }

        // tiny holes, to increase the perimeter of the bottom bit and make it
        // stronger
        translate([-tiny(),0,flex_dims().z]){
            cube([2*tiny(), 10, 4]);
        }
        // cut off at the bottom
        mirror([0,0,1]){
            cylinder(r=999,h=999,$fn=4);
        }
    }
}
module actuator_end_cutout(lever_tip=3-0.5)
[Source]
module actuator_end_cutout(lever_tip=3-0.5 ){
    // This shape cuts off the end of an actuator, leaving a thin strip to
    // connect to the actuator column (the flexure).
    sequential_hull(){
        translate([-999,-flex_dims().y/2,flex_dims().z]){
            cube([2,2,2]*999);
        }
        translate([-999,-flex_dims().y/2,flex_dims().z+lever_tip]){
            cube([2,2,2]*999);
        }
        translate([-999,-flex_dims().y/2-999,flex_dims().z+999]){
            cube([2,2,2]*999);
        }
    }
}
module nut_seat_silhouette(offset=0)
[Source]
module nut_seat_silhouette(offset=0){
    // a (2D) shape made from the convex hull of two circles od radius r
    // we don't actually build it like that though, as the hull is a slow operation.
    r=actuator_housing_xy_size().y/2;
    dx=actuator_housing_xy_size().x-actuator_housing_xy_size().y;
    union(){
        reflect([1,0]){
            translate([dx/2,0]){
                circle(r=r+offset);
            }
        }
        square([dx,2*(r+offset)], center=true);
    }
}
module nut_seat_void(h=1, tilt=0, center=true)
[Source]
module nut_seat_void(h=1, tilt=0, center=true){
    // Inside of the actuator column housing (should be subtracted
    // h is the height of the top (excluding nut hole)
    // center=true will cause it to punch through the bottom.
    // This ensures enough clearance to let the actuator column move.
    rotate_x(tilt){
        intersection(){
            linear_extrude(999,center=center){
                nut_seat_silhouette(offset=-actuator_wall_t());
            }
            translate_z(h){
                rotate(90){
                    hole_from_bottom(actuator_nut_size()*1.1/2, h=999, base_w=999);
                }
            }
        }
    }
}
module screw_seat_shell(h=1, tilt=0)
[Source]
module screw_seat_shell(h=1, tilt=0){
    // Outside of the actuator column housing - this is the structure that
    // the gear sits on top of.  It needs to be hollowed out before use
    // (see screw_seat)
    // Create a slightly over double height column and cut off bottom
    double_h = (h+2)*2;
    difference(){
        rotate_x(tilt){
            hull(){
                linear_extrude(double_h-3, center=true){
                    nut_seat_silhouette();
                }
                linear_extrude(double_h, center=true){
                    nut_seat_silhouette(offset=-2);
                }
            }
        }
        mirror([0,0,1]){
            //ground
            cylinder(r=999,h=999,$fn=8);
        }
    }
}
module motor_lugs(h, tilt=0, angle=0, brackets=true)
[Source]
module motor_lugs(h, tilt=0, angle=0, brackets=true){
    screw_pos = motor_screw_pos(h);
    // lugs to mount a micro geared stepper motor on a screw_seat.
    screw_r = sqrt(pow(screw_pos.x,2)+pow(screw_pos.y,2));
    lug_cyl_r = 5;
    rotate_x(tilt){
        rotate(angle){
            reflect_x(){
                difference(){
                    union(){
                        translate(screw_pos-[0,0,motor_lug_h()]){
                            cylinder(r=lug_cyl_r,h=motor_lug_h()+0.6);
                        }
                        if (brackets){
                            // brackets to connect the motor screw cylinders to the screw_seat
                            hull(){
                                translate(screw_pos-[0,0,motor_lug_h()]){
                                    //hemisphere
                                    difference(){
                                        sphere(r=lug_cyl_r);
                                        translate_z(99/2){
                                            cube(99,center=true);
                                        }
                                    }
                                    //Add a thin cylinder, because an STL sphere is smaller at
                                    //the equator than a cylinder of the same r
                                    translate_z(-tiny()){
                                        cylinder(r=lug_cyl_r, h=tiny());
                                    }
                                }
                                translate_z(screw_pos.z-screw_r-motor_lug_h()){
                                    cylinder(r=4,h=screw_r-5);
                                }
                            }
                        }
                    }
                    //space for gears
                    translate_z(h){
                        cylinder(r1=8,r2=16,h=2+tiny());
                    }
                    translate_z(h+2){
                        cylinder(r=16, h=999);
                    }
                    //hollow inside of the structure
                    rotate(-angle){
                        nut_seat_void(h=h, tilt=0);
                    }
                    //mounting screws
                    translate(screw_pos){
                        //imprint for the metal lugs on the motor 
                        rotate_z(90){
                            translate_y(5){
                                cyl_slot(r=7.1/2, h=100, dy=10, $fn=24);
                            }
                        }
                        translate_z(-9){
                            m3_nut_trap_with_shaft(slot_angle=90,deep_shaft=3+tiny(),chamfer_offset=3.5);
                        }
                    }
                }
            }
        }
    }
}
module screw_seat(
    params,
    h,
    travel,
    tilt=0,
    extra_entry_h=7,
    include_motor_lugs=undef,
    lug_angle=0,
    label="",
    lug_brackets=true
)
[Source]
module screw_seat(params, h, travel, tilt=0, extra_entry_h=7, include_motor_lugs=undef, lug_angle=0, label="", lug_brackets=true){
    // This forms a hollow column, usually built around an actuator_column to
    // support the screw (see screw_seat_shell)

    create_motor_lugs = if_undefined_set_default(include_motor_lugs,
                                                 key_lookup("include_motor_lugs", params));

    entry_h = extra_entry_h + travel; //ensure the actuator can move
    nut_slot_z = h-actuator_nut_size()-1.5-actuator_nut_slot_size().z;
    difference(){
        union(){
            screw_seat_shell(h=h + travel, tilt=tilt);

            if(create_motor_lugs){
                rotate(180){
                    motor_lugs(h=h + travel, angle=lug_angle, tilt=-tilt, brackets=lug_brackets);
                }
            }
            if(len(label) > 0){
                rotate_x(tilt){
                    translate([0, actuator_housing_xy_size().y/2, nut_slot_z - 2]){
                        rotate_x(90){
                            linear_extrude(1, center=true){
                                mirror([1,0]){
                                    text(label, size=10, font="Sans", halign="center", valign="top");
                                }
                            }
                        }
                    }
                }
            }
        }
        //hollow out the inside
        nut_seat_void(h=h + travel, tilt=tilt);

        //allow the actuator to poke in
        edge_y = actuator_housing_xy_size().y/2;
        translate_y(-edge_y){
            rotate_x(tilt){
                sparse_matrix_transform(zy=sin(-tilt)){
                    cube([actuator_entry_width(), edge_y, entry_h*2], center=true);
                }
            }
        }

        //entrance slot for nut
        rotate_x(tilt){
            translate_z(nut_slot_z){
                nut_trap_and_slot(actuator_nut_size(), actuator_nut_slot_size() + [0,0,0.3]);
            }
        }
    }
}
module screw_seat_outline(h=999, adjustment=0, center=false, tilt=0)
[Source]
module screw_seat_outline(h=999,adjustment=0,center=false,tilt=0){
    // The bottom of a screw seat
    rotate_x(tilt){
        linear_extrude(h,center=center){
            nut_seat_silhouette(offset=adjustment);
        }
    }
}