diff --git a/3d/28byj-48.scad b/3d/28byj-48.scad index 3af5ff0a..06fc371d 100644 --- a/3d/28byj-48.scad +++ b/3d/28byj-48.scad @@ -16,6 +16,9 @@ 28byj48_chassis_radius = 28/2; 28byj48_chassis_height = 19; +// Clearance between the motor chassis and the outside right wall of the previous module +28byj48_chassis_height_clearance = 1.4; + 28byj48_shaft_offset = 8; 28byj48_mount_outer_radius = 3.5; 28byj48_mount_hole_radius = 4.2/2; diff --git a/3d/common.scad b/3d/common.scad new file mode 100644 index 00000000..a2304bfc --- /dev/null +++ b/3d/common.scad @@ -0,0 +1,29 @@ +// MDF, .120in nominal +// https://www.ponoko.com/materials/mdf-fiberboard +thickness = 3.0; + +etch_depth = 0.1; // for render +eps=.01; +num_flaps = 40; + +// Rendering Colors +assembly_color = [0.76, 0.60, 0.42]; // MDF, "c1996b" +etch_color = [0, 0, 0]; // black, "000000" + + +assembly_color1 = color_multiply(assembly_color, [1.161, 1.157, 1.157, 1.0]); // "e1b17c" with MDF +assembly_color2 = color_multiply(assembly_color, [0.897, 0.895, 0.895, 1.0]); // "ae8960" with MDF +assembly_color3 = color_multiply(assembly_color, [0.547, 0.542, 0.540, 1.0]); // "6a533a" with MDF +assembly_color4 = color_multiply(assembly_color, [0.268, 0.268, 0.271, 1.0]); // "34291d" with MDF + +// multiply two equal matricies by each element, limiting to a max of 1.0 +function color_multiply(x, y) = + [ for(j=[0:len(x) - 1]) min(x[j] * y[j], 1.0) ]; + + +module enclosure_etch_style() { + color(etch_color) + translate([0, 0, thickness]) + linear_extrude(height=etch_depth) + children(); +} diff --git a/3d/flap_fonts.scad b/3d/flap_fonts.scad index 4e4825a0..174b45e1 100644 --- a/3d/flap_fonts.scad +++ b/3d/flap_fonts.scad @@ -15,7 +15,6 @@ */ use; -include; letter_font = "RobotoCondensed"; letter_height = 0.75; // aspect, 0-1 diff --git a/3d/font_generator.scad b/3d/font_generator.scad index 4a6dc26d..08c2bdd6 100644 --- a/3d/font_generator.scad +++ b/3d/font_generator.scad @@ -14,7 +14,7 @@ limitations under the License. */ -include; +include; use; use; diff --git a/3d/m4_dimensions.scad b/3d/m4_dimensions.scad index d6177169..a48cebcd 100644 --- a/3d/m4_dimensions.scad +++ b/3d/m4_dimensions.scad @@ -14,14 +14,4 @@ limitations under the License. */ -// M4 bolts -m4_hole_diameter = 4.5; -m4_bolt_length = 12; -m4_button_head_diameter = 7.6 + .2; -m4_button_head_length = 2.2 + .2; -m4_nut_width_flats = 7 + .2; -m4_nut_width_corners = 7/cos(180/6); -m4_nut_width_corners_padded = m4_nut_width_corners + .2; -m4_nut_length = 3.2; -m4_nut_length_padded = m4_nut_length + .2; diff --git a/3d/parts/connector.scad b/3d/parts/connector.scad new file mode 100644 index 00000000..3341af7c --- /dev/null +++ b/3d/parts/connector.scad @@ -0,0 +1,57 @@ +/* + Copyright 2015-2020 Scott Bezek and the splitflap contributors + + Licensed under the Apache License, Version 2.0 (the "License"); + you may not use this file except in compliance with the License. + You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + + Unless required by applicable law or agreed to in writing, software + distributed under the License is distributed on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + See the License for the specific language governing permissions and + limitations under the License. +*/ + +include; +include; +include<../common.scad>; + +connector_bracket_length_outer = 14; +connector_bracket_length_inner = side_tab_width * 2 - m4_button_head_diameter/2; +connector_bracket_thickness = captive_nut_inset - thickness - 0.2; +connector_bracket_width = enclosure_width - enclosure_wall_to_wall_width + thickness*2 + connector_bracket_thickness*2; +connector_bracket_overlap = 4; +connector_bracket_clearance = 0.40; +connector_bracket_depth_clearance = 0.20; + + +module connector_bracket_2d() { + difference() { + square([connector_bracket_width, connector_bracket_length_outer]); + translate([connector_bracket_thickness, -eps]) { + square([connector_bracket_width - connector_bracket_thickness*2, connector_bracket_length_outer - connector_bracket_length_inner + eps]); + } + translate([connector_bracket_thickness - connector_bracket_clearance/2, -eps]) { + square([thickness + connector_bracket_clearance, connector_bracket_length_outer - connector_bracket_overlap + connector_bracket_depth_clearance + eps]); + } + translate([connector_bracket_width - connector_bracket_thickness - thickness - connector_bracket_clearance/2, -eps]) { + square([thickness + connector_bracket_clearance, connector_bracket_length_outer - connector_bracket_overlap + connector_bracket_depth_clearance + eps]); + } + } +} + +module connector_bracket() { + linear_extrude(height=thickness) { + connector_bracket_2d(); + } +} + +module connector_bracket_side_holes() { + // overlap slot + translate([enclosure_vertical_inset - thickness - connector_bracket_clearance/2, -connector_bracket_overlap]) { + square([thickness + connector_bracket_clearance, connector_bracket_overlap + eps]); + } +} + diff --git a/3d/parts/enclosure.scad b/3d/parts/enclosure.scad new file mode 100644 index 00000000..6095ab3f --- /dev/null +++ b/3d/parts/enclosure.scad @@ -0,0 +1,419 @@ +/* + Copyright 2015-2020 Scott Bezek and the splitflap contributors + + Licensed under the Apache License, Version 2.0 (the "License"); + you may not use this file except in compliance with the License. + You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + + Unless required by applicable law or agreed to in writing, software + distributed under the License is distributed on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + See the License for the specific language governing permissions and + limitations under the License. +*/ + +include; +include; + +// Enclosure tabs: front/back +enclosure_tab_clearance = 0.10; + +enclosure_left_zip_side_inset = 5.0; // inset from left for the bottom zip tie holes, edge to outside edge +enclosure_left_zip_bottom_inset = 22.5; // inset from bottom for the bottom zip tie holes, edge to group center + +enclosure_left_zip_top_inset = 22.5; // inset from top for the top zip tie holes, edge to group center + +// Radius where flaps are expected to flap in their *most collapsed* (90 degree) state +exclusion_radius = sqrt(flap_height*flap_height + flap_pitch_radius*flap_pitch_radius); +// Radius where flaps are expected to flap in their *most extended* state +outer_exclusion_radius = flap_pitch_radius + flap_height + 2; +front_forward_offset = flap_pitch_radius + flap_thickness/2; + +// Width measured from the outside of the walls +enclosure_wall_to_wall_width = thickness + spool_width_slop/2 + spool_width_clearance + spool_width_slop/2 + max(28byj48_mount_bracket_height + m4_button_head_length, 4 + 28byj48_mount_bracket_height - spool_width_slop/2) + thickness; + +// Width of the front panel +enclosure_width = enclosure_wall_to_wall_width + 28byj48_chassis_height + 28byj48_chassis_height_clearance - thickness - 28byj48_mount_bracket_height; +enclosure_horizontal_inset = (enclosure_width - enclosure_wall_to_wall_width)/2; +front_window_upper_base = (flap_height - flap_pin_width/2); +front_window_overhang = 3; +front_window_upper = front_window_upper_base - front_window_overhang; +front_window_lower = sqrt(outer_exclusion_radius*outer_exclusion_radius - front_forward_offset*front_forward_offset); +front_window_slop = 0; +front_window_width = spool_width_slop + spool_width_clearance + front_window_slop; +front_window_right_inset = thickness - front_window_slop/2; + + +enclosure_vertical_margin = 10; // gap between top/bottom of flaps and top/bottom of enclosure +enclosure_vertical_inset = max(thickness*1.5, m4_nut_width_corners_padded/2); // distance from top of sides to top of the top piece +enclosure_height_upper = exclusion_radius + enclosure_vertical_margin + thickness + enclosure_vertical_inset; +enclosure_height_lower = flap_pitch_radius + flap_height + enclosure_vertical_margin + thickness + enclosure_vertical_inset; +enclosure_height = enclosure_height_upper + enclosure_height_lower; + +enclosure_horizontal_rear_margin = thickness; // minumum distance between the farthest feature and the rear +enclosure_length = front_forward_offset + 28byj48_mount_center_offset + m4_hole_diameter/2 + enclosure_horizontal_rear_margin; +enclosure_length_right = front_forward_offset + m4_hole_diameter/2 + 2; + +enclosure_indicator_inset = 3.0; // inset on both X and Y +enclosure_indicator_size = 1.75; // symbol size +enclosure_indicator_arrow_width = 2.25; +enclosure_indicator_arrow_height = enclosure_indicator_arrow_width * 2; +enclosure_indicator_position_y = (enclosure_height - enclosure_vertical_inset - thickness) - enclosure_indicator_inset; + +backstop_bolt_vertical_offset = - (exclusion_radius + outer_exclusion_radius)/2; +backstop_bolt_forward_range = 14; + +mounting_hole_inset = m4_button_head_diameter/2 + 2; + +side_tab_width = enclosure_length_right / 4; +side_tab_width_fraction = 0.5; +captive_nut_inset=6; +num_front_tabs = 2; +front_tab_width = (enclosure_wall_to_wall_width - 2*thickness) / (num_front_tabs*2 - 1); + + +module front_tabs_negative() { + for (i = [0 : num_front_tabs-1]) { + translate([thickness + (i*2+0.5) * front_tab_width, 0, 0]) + square([front_tab_width + enclosure_tab_clearance, thickness + enclosure_tab_clearance], center=true); + } + for (i = [0 : num_front_tabs-2]) { + translate([thickness + (i*2+1.5) * front_tab_width, 0, 0]) + circle(r=m4_hole_diameter/2, $fn=30); + } +} + + + +module enclosure_front() { + linear_extrude(height=thickness) { + difference() { + translate([-enclosure_horizontal_inset, 0, 0]) { + square([enclosure_width, enclosure_height]); + } + + // Viewing window cutout + translate([front_window_right_inset, enclosure_height_lower - front_window_lower]) + square([front_window_width, front_window_lower + front_window_upper]); + + // Front lower tabs + translate([0, thickness * 0.5 + enclosure_vertical_inset, 0]) + front_tabs_negative(); + + // Front upper tabs + translate([0, enclosure_height - thickness * 0.5 - enclosure_vertical_inset, 0]) + front_tabs_negative(); + } + } +} + +module enclosure_front_etch() { + // alignment indicator, left side (triangle) + enclosure_etch_style() + translate([enclosure_wall_to_wall_width - thickness - enclosure_indicator_inset, enclosure_indicator_position_y]) + triangle(enclosure_indicator_size, center=true); + + // alignment indicator, right side (circle) + enclosure_etch_style() + translate([thickness + enclosure_indicator_inset, enclosure_indicator_position_y]) + circle(r=enclosure_indicator_size/2, $fn=60); + + // position indicator, 'up' arrow + enclosure_etch_style() + translate([enclosure_width - enclosure_horizontal_inset * 1.5, enclosure_height - enclosure_indicator_arrow_height/2 - enclosure_indicator_inset]) + arrow([enclosure_indicator_arrow_width, enclosure_indicator_arrow_height], center=true); +} + +module alignment_bar() { + color(assembly_color1) + translate([enclosure_width - enclosure_horizontal_inset, -enclosure_length_right + front_forward_offset - alignment_bar_diameter/2, -enclosure_height_lower + alignment_bar_diameter/2]) + rotate([0, -90, 0]) + linear_extrude(height=enclosure_width * len(render_letters)) + circle(r=alignment_bar_diameter/2, $fn=60); +} + +module enclosure_left() { + linear_extrude(height=thickness) { + difference() { + square([enclosure_height, enclosure_length]); + translate([enclosure_height_lower, enclosure_length - front_forward_offset, 0]) + circle(r=m4_hole_diameter/2, $fn=30); + + translate([enclosure_height_lower, enclosure_length - front_forward_offset]) { + rotate([0, 0, 90]) { + motor_mount(); + } + } + + // bottom side tabs + translate([thickness * 0.5 + enclosure_vertical_inset, enclosure_length, 0]) + mirror([0, 1, 0]) + side_tabs_negative(hole_sizes=[1]); + + // top side tabs + translate([enclosure_height - thickness * 0.5 - enclosure_vertical_inset, enclosure_length, 0]) + mirror([0, 1, 0]) + side_tabs_negative(hole_sizes=[1]); + + // Connector bracket cuts + translate([enclosure_height, enclosure_length]) { + mirror([1, 0, 0]) { + connector_bracket_side_holes(); + } + } + translate([0, enclosure_length]) { + connector_bracket_side_holes(); + } + + + // PCB mounting holes + translate([enclosure_height_lower - magnet_hole_offset - pcb_hole_to_sensor_y, enclosure_length - front_forward_offset - pcb_hole_to_sensor_x]) { + rotate([180, 0, 0]) { + rotate([0, 0, -90]) { + pcb_cutouts(); + } + } + } + + // Zip tie holes, sensor (leading to bottom) + translate([enclosure_left_zip_bottom_inset, zip_tie_height/2 + enclosure_left_zip_side_inset, 0]) + zip_tie_holes(); + + // Zip tie holes, motor (leading to top) + translate([enclosure_height - enclosure_left_zip_top_inset, enclosure_length - front_forward_offset]) + rotate([0, 0, 90]) // cable channel facing 'up' + zip_tie_holes(); + + // Alignment bar cutout + translate([0, alignment_bar_center]) { + union() { + // Cutout + translate([alignment_bar_cutout_width/2, 0]) + circle(r=alignment_bar_cutout_width/2, $fn=40); + square([alignment_bar_cutout_width, alignment_bar_cutout_width], center=true); + + // Front-side fillet + // translate([0, alignment_bar_cutout_width/2, 0]) + // fillet_tool(r=alignment_bar_fillet_radius, overlap=1, $fn=40); + + // Back-side fillet + translate([0, -alignment_bar_cutout_width/2, 0]) + mirror([0, 1, 0]) + fillet_tool(r=alignment_bar_fillet_radius, $fn=40); + } + } + } + } +} + +module enclosure_left_etch() { + // alignment indicator (triangle) + enclosure_etch_style() + translate([enclosure_indicator_position_y, enclosure_length - enclosure_indicator_inset]) + rotate([0, 0, -90]) + triangle(enclosure_indicator_size, center=true); +} + +module enclosure_right() { + linear_extrude(height=thickness) { + difference() { + square([enclosure_height, enclosure_length_right]); + translate([enclosure_height_upper, enclosure_length_right - front_forward_offset, 0]) + circle(r=m4_hole_diameter/2, $fn=30); + + // backstop bolt slot + translate([enclosure_height_upper - backstop_bolt_vertical_offset, enclosure_length_right - front_forward_offset, 0]) { + backstop_bolt_slot(radius = m4_hole_diameter/2); + } + + // top side tabs + translate([0.5*thickness + enclosure_vertical_inset, enclosure_length_right, 0]) + mirror([0, 1, 0]) + side_tabs_negative(hole_sizes=[1]); + + // bottom side tabs + translate([enclosure_height - 0.5*thickness - enclosure_vertical_inset, enclosure_length_right, 0]) + mirror([0, 1, 0]) + side_tabs_negative(hole_sizes=[1]); + + // Connector bracket cuts + translate([enclosure_height, enclosure_length_right]) { + mirror([1, 0, 0]) { + connector_bracket_side_holes(); + } + } + translate([0, enclosure_length_right]) { + connector_bracket_side_holes(); + } + } + } +} + +module enclosure_right_etch() { + // alignment indicator (circle) + enclosure_etch_style() + translate([enclosure_height - enclosure_indicator_position_y, enclosure_length_right - enclosure_indicator_inset]) + circle(r=enclosure_indicator_size/2, $fn=60); +} + +module front_back_tabs() { + for (i = [0 : 2 : num_front_tabs*2-2]) { + translate([i * front_tab_width, -eps, 0]) + square([front_tab_width, thickness + eps]); + } +} + +module side_tabs(tabs) { + for (i = [0 : 2 : tabs*2-2]) { + translate([-eps, i * side_tab_width + side_tab_width * (1 - side_tab_width_fraction)/2, 0]) + square([thickness + eps, side_tab_width * side_tab_width_fraction]); + } +} + +module front_back_captive_nuts() { + for (i = [0 : 2 : num_front_tabs-1]) { + translate([(i*2 + 1.5) * front_tab_width, -thickness, 0]) + m4_captive_nut(); + } +} + +module side_captive_nuts(hole_types=[]) { + for (i = [0 : len(hole_types)-1]) { + hole_type = hole_types[i]; + translate([-thickness, (i*2 + 1.5) * side_tab_width, 0]) { + rotate([0, 0, -90]) { + if (hole_type == 2) { + } else if (hole_type == 1) { + m4_captive_nut(); + } + } + } + } +} + + +module enclosure_top() { + // note, this is flipped upside down (around the x axis) when assembled so the clean side faces out + linear_extrude(height = thickness) { + translate([thickness, 0, 0]) { + difference() { + union() { + square([enclosure_wall_to_wall_width - 2 * thickness, enclosure_length_right]); + + // front tabs + mirror([0, 1, 0]) + front_back_tabs(); + + // left tabs + translate([enclosure_wall_to_wall_width - 2 * thickness, thickness, 0]) + side_tabs(2); + + // right tabs + mirror([1, 0, 0]) + translate([0, thickness, 0]) + side_tabs(2); + } + + // front captive nuts + front_back_captive_nuts(); + + // right captive nuts + translate([0, thickness, 0]) + side_captive_nuts(hole_types = [1]); + + // left captive nuts + translate([enclosure_wall_to_wall_width - 2 * thickness, thickness, 0]) + mirror([1, 0, 0]) + side_captive_nuts(hole_types = [1]); + + // mounting hole + translate([(enclosure_wall_to_wall_width - 2 * thickness)/2, enclosure_length_right - mounting_hole_inset]) { + circle(r=m4_hole_diameter/2, $fn=30); + } + } + } + } +} + +module enclosure_bottom() { + linear_extrude(height = thickness) { + translate([thickness, 0, 0]) { + difference() { + union() { + square([enclosure_wall_to_wall_width - 2 * thickness, enclosure_length_right]); + + // front tabs + translate([0, enclosure_length_right, 0]) + front_back_tabs(); + + // left tabs + translate([enclosure_wall_to_wall_width - 2 * thickness, enclosure_length_right - thickness, 0]) + mirror([0, 1, 0]) + side_tabs(2); + + // right tabs + translate([0, enclosure_length_right - thickness, 0]) + mirror([0, 1, 0]) + mirror([1, 0, 0]) + side_tabs(2); + } + + // front captive nuts + translate([0, enclosure_length_right, 0]) + mirror([0,1,0]) + front_back_captive_nuts(); + + // right captive nuts + translate([0, enclosure_length_right - thickness, 0]) + mirror([0, 1, 0]) + side_captive_nuts(hole_types = [1]); + + // left captive nuts + translate([enclosure_wall_to_wall_width - 2 * thickness, enclosure_length_right - thickness, 0]) + mirror([0, 1, 0]) + mirror([1, 0, 0]) + side_captive_nuts(hole_types = [1]); + + // mounting hole + translate([(enclosure_wall_to_wall_width - 2 * thickness)/2, mounting_hole_inset]) { + circle(r=m4_hole_diameter/2, $fn=30); + } + } + } + } +} + +module enclosure_bottom_etch() { + enclosure_etch_style() + translate([captive_nut_inset + m4_nut_length + 1, 1, thickness]) { + text_label([str("rev. ", render_revision), render_date, "github.com/scottbez1/splitflap"]); + } +} + +module side_tabs_negative(hole_sizes=[], extend_last_tab=false) { + for (i = [0 : len(hole_sizes)]) { + length = (extend_last_tab && i == len(hole_sizes)) ? side_tab_width * side_tab_width_fraction + eps : side_tab_width * side_tab_width_fraction; + translate([0, thickness + (i*2) * side_tab_width + side_tab_width * (1 - side_tab_width_fraction)/2 + length/2, 0]) + square([thickness + enclosure_tab_clearance, length + enclosure_tab_clearance], center=true); + } + for (i = [0 : len(hole_sizes) - 1]) { + hole_size = hole_sizes[i]; + if (hole_size > 0) { + bolt_head_hole = hole_size == 2; + translate([0, thickness + (i*2 + 1.5) * side_tab_width, 0]) + circle(r=(bolt_head_hole ? m4_button_head_diameter : m4_hole_diameter)/2, $fn=30); + } + } +} + +module backstop_bolt_slot(radius) { + hull() { + circle(r=radius, $fn=15); + translate([0, backstop_bolt_forward_range]) { + circle(r=radius, $fn=15); + } + } +} + diff --git a/3d/parts/flap.scad b/3d/parts/flap.scad new file mode 100644 index 00000000..a36e0455 --- /dev/null +++ b/3d/parts/flap.scad @@ -0,0 +1,107 @@ +/* + Copyright 2018 Scott Bezek and the splitflap contributors + + Licensed under the Apache License, Version 2.0 (the "License"); + you may not use this file except in compliance with the License. + You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + + Unless required by applicable law or agreed to in writing, software + distributed under the License is distributed on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + See the License for the specific language governing permissions and + limitations under the License. +*/ + +include<../flap_fonts.scad>; + +flap_width = 54; +flap_height = 43; +flap_thickness = 30 / 1000 * 25.4; // 30 mil +flap_corner_radius = 3.1; // 2.88-3.48mm (used just for display) +flap_width_slop = 0.5; // amount of slop of the flap side to side between the 2 spools + +flap_notch_height_auto = false; +flap_notch_height_default = 15; +flap_notch_depth = 3.2; + +flap_notch_height = (flap_notch_height_auto == true) ? sqrt(spool_outer_radius*spool_outer_radius - flap_pitch_radius*flap_pitch_radius) : flap_notch_height_default; + +flap_pin_width = 1.4; +flap_color = [1, 1, 1]; // white, "ffffff" + +eps=.01; + +flap_hole_radius = (flap_pin_width + 1) / 2; +flap_hole_separation = 1; // additional spacing between hole edges +flap_gap = (flap_hole_radius * 2 - flap_pin_width) + flap_hole_separation; +letter_color = color_invert(flap_color); // inverse of the flap color, for contrast + +// flap(); + +// inverts a color matrix by subtracting the input channel values from 1.0 +function color_invert(x) = + [ for(j=[0:len(x) - 1]) (1.0 - x[j]) ]; + +module flap_2d(cut_tabs = true) { + translate([0, -flap_pin_width/2, 0]) + difference() { + union() { + square([flap_width, flap_height - flap_corner_radius]); + + // rounded corners + hull() { + translate([flap_corner_radius, flap_height - flap_corner_radius]) + circle(r=flap_corner_radius, $fn=40); + translate([flap_width - flap_corner_radius, flap_height - flap_corner_radius]) + circle(r=flap_corner_radius, $fn=40); + } + } + // spool tabs + if(cut_tabs) { + translate([-eps, flap_pin_width]) + square([eps + flap_notch_depth, flap_notch_height]); + translate([flap_width - flap_notch_depth, flap_pin_width]) + square([eps + flap_notch_depth, flap_notch_height]); + } + } +} + +module flap() { + color(flap_color) + translate([0, 0, -flap_thickness/2]) + linear_extrude(height=flap_thickness) { + flap_2d(); + } +} + +module draw_letter(letter) { + translate([0, -flap_height * letter_height, 0]) // valign compensation + scale([letter_width, 1, 1]) + translate([letter_offset_x, letter_offset_y]) + text(text=letter, size=flap_height * letter_height * 2, font=letter_font, halign="center"); +} + +module flap_letter(letter, half = 0) { + color(letter_color) + translate([0, 0, flap_thickness/2 + eps]) + linear_extrude(height=0.1, center=true) { + if (half != 0) { // trimming to top (1) or bottom (2) + intersection() { + flap_2d(); // limit to bounds of flap + translate([flap_width/2, -flap_pin_width/2, 0]) { + rotation = (half == 2) ? -180 : 0; // flip upside-down for bottom + gap_comp = (letter_gap_comp == true) ? -flap_gap/2 : 0; + translate([0, gap_comp, 0]) + rotate([0,0,rotation]) + draw_letter(letter); + } + } + } else { + translate([flap_width/2, -flap_pin_width/2 - flap_gap/2, 0]) + draw_letter(letter); + } + } +} + diff --git a/3d/parts/hardware.scad b/3d/parts/hardware.scad new file mode 100644 index 00000000..80c58a89 --- /dev/null +++ b/3d/parts/hardware.scad @@ -0,0 +1,103 @@ +/* + Copyright 2015-2020 Scott Bezek and the splitflap contributors + + Licensed under the Apache License, Version 2.0 (the "License"); + you may not use this file except in compliance with the License. + You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + + Unless required by applicable law or agreed to in writing, software + distributed under the License is distributed on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + See the License for the specific language governing permissions and + limitations under the License. +*/ +include<../common.scad>; + + +zip_tie_height = 3.0; // height of the zip-tie hole +zip_tie_width = 2.0; // width of the zip-tie holes +zip_tie_spacing = 6.5; // spacing between each zip-tie hole, inside edges +zip_tie_fillet = 0.5; // radius of the ounded zip-tie hole corners + + +// M4 bolts +m4_hole_diameter = 4.5; +m4_bolt_length = 12; +m4_button_head_diameter = 7.6 + .2; +m4_button_head_length = 2.2 + .2; +m4_nut_width_flats = 7 + .2; +m4_nut_width_corners = 7/cos(180/6); +m4_nut_width_corners_padded = m4_nut_width_corners + .2; +m4_nut_length = 3.2; +m4_nut_length_padded = m4_nut_length + .2; + +hardware_color = [0.75, 0.75, 0.8]; // steel, "bfbfcc" +bolt_color = hardware_color; +nut_color = color_multiply(hardware_color, [0.933, 0.933, 0.9, 1.0]); // "b2b2b7" with steel + +module standard_m4_bolt(nut_distance=-1) { + if (render_bolts) { + color(bolt_color) + roughM4_7380(10); + if (nut_distance >= 0) { + color(nut_color) { + translate([0, 0, nut_distance]) { + linear_extrude(m4_nut_length) { + difference() { + circle(r=m4_nut_width_corners/2, $fn=6); + circle(r=m4_hole_diameter/2, $fn=20); + } + } + } + } + } + } +} + +// ##### CAPTIVE NUT NEGATIVE ##### + +// Centered in the x dimension +module captive_nut(bolt_diameter, bolt_length, nut_width, nut_length, nut_inset) { + union() { + translate([-bolt_diameter/2, 0, 0]) + square([bolt_diameter, bolt_length]); + translate([-nut_width/2, nut_inset, 0]) + square([nut_width, nut_length]); + } +} +module m4_captive_nut(bolt_length=m4_bolt_length) { + captive_nut(m4_hole_diameter, bolt_length + 1, m4_nut_width_flats, m4_nut_length_padded, captive_nut_inset); +} + + +module zip_tie_holes() { + spacing = (zip_tie_spacing + zip_tie_width)/2; + + translate([-spacing, 0, 0]) + rounded_square([zip_tie_width, zip_tie_height], center=true, r=zip_tie_fillet, $fn=30); + translate([spacing, 0, 0]) + rounded_square([zip_tie_width, zip_tie_height], center=true, r=zip_tie_fillet, $fn=30); +} + +module front_back_captive_nuts() { + for (i = [0 : 2 : num_front_tabs-1]) { + translate([(i*2 + 1.5) * front_tab_width, -thickness, 0]) + m4_captive_nut(); + } +} + +module side_captive_nuts(hole_types=[]) { + for (i = [0 : len(hole_types)-1]) { + hole_type = hole_types[i]; + translate([-thickness, (i*2 + 1.5) * side_tab_width, 0]) { + rotate([0, 0, -90]) { + if (hole_type == 2) { + } else if (hole_type == 1) { + m4_captive_nut(); + } + } + } + } +} diff --git a/3d/flap_dimensions.scad b/3d/parts/magnet.scad similarity index 63% rename from 3d/flap_dimensions.scad rename to 3d/parts/magnet.scad index 3267edb9..6f405007 100644 --- a/3d/flap_dimensions.scad +++ b/3d/parts/magnet.scad @@ -1,5 +1,5 @@ /* - Copyright 2018 Scott Bezek and the splitflap contributors + Copyright 2015-2020 Scott Bezek and the splitflap contributors Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. @@ -14,14 +14,9 @@ limitations under the License. */ -flap_width = 54; -flap_height = 43; -flap_thickness = 30 / 1000 * 25.4; // 30 mil -flap_corner_radius = 3.1; // 2.88-3.48mm (used just for display) - -flap_notch_height_auto = false; -flap_notch_height_default = 15; -flap_notch_depth = 3.2; - -flap_pin_width = 1.4; +include ; +magnet_diameter = 4; +magnet_hole_clearance = -0.07; // interference fit +magnet_hole_radius = (magnet_diameter + magnet_hole_clearance)/2; +magnet_hole_offset = (spool_strut_exclusion_radius + flap_pitch_radius)/2; diff --git a/3d/parts/motor.scad b/3d/parts/motor.scad new file mode 100644 index 00000000..52adba55 --- /dev/null +++ b/3d/parts/motor.scad @@ -0,0 +1,58 @@ +/* + Copyright 2015-2016 Scott Bezek and the splitflap contributors + + Licensed under the Apache License, Version 2.0 (the "License"); + you may not use this file except in compliance with the License. + You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + + Unless required by applicable law or agreed to in writing, software + distributed under the License is distributed on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + See the License for the specific language governing permissions and + limitations under the License. +*/ + +include<../28byj-48.scad>; +include; + +motor_shaft_under_radius = 0.08; // interference fit +motor_slop_radius = 3; +motor_mount_hole_radius = m4_hole_diameter/2; +motor_backpack_extent = 28byj48_backpack_extent + 2; // Add 2mm to make sure there's room for the wires +motor_hole_slop = 1; +motor_window_radius = 5; + +// holes for 28byj-48 motor, centered around motor shaft +module motor_mount() { + circle(r=28byj48_shaft_radius+motor_slop_radius, $fn=30); + translate([-28byj48_mount_center_offset, -8]) { + circle(r=motor_mount_hole_radius, $fn=30); + } + translate([28byj48_mount_center_offset, -8]) { + circle(r=motor_mount_hole_radius, $fn=30); + } + + hull() { + x = -28byj48_chassis_radius - motor_hole_slop/2 + motor_window_radius; + y = [-28byj48_shaft_offset - motor_backpack_extent - motor_hole_slop/2 + motor_window_radius, + -28byj48_shaft_offset + 28byj48_chassis_radius + motor_hole_slop/2 - motor_window_radius]; + + translate([ x, y[0], 0]) circle(r=motor_window_radius, $fn=40); + translate([-x, y[1], 0]) circle(r=motor_window_radius, $fn=40); + translate([-x, y[0], 0]) circle(r=motor_window_radius, $fn=40); + translate([ x, y[1], 0]) circle(r=motor_window_radius, $fn=40); + } +} + +// double-flatted motor shaft of 28byj-48 motor (2D) +module motor_shaft() { + union() { + intersection() { + circle(r=28byj48_shaft_radius-motor_shaft_under_radius, $fn=50); + square([28byj48_shaft_radius*2, 3], center=true); + } + square([28byj48_shaft_radius/3, 28byj48_shaft_radius*4], center=true); + } +} diff --git a/3d/parts/spool.scad b/3d/parts/spool.scad new file mode 100644 index 00000000..1b7148d0 --- /dev/null +++ b/3d/parts/spool.scad @@ -0,0 +1,220 @@ +/* + Copyright 2015-2016 Scott Bezek and the splitflap contributors + + Licensed under the Apache License, Version 2.0 (the "License"); + you may not use this file except in compliance with the License. + You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + + Unless required by applicable law or agreed to in writing, software + distributed under the License is distributed on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + See the License for the specific language governing permissions and + limitations under the License. +*/ + +include<../common.scad>; +include; +include; +include; + +thickness = 3.0; + +spool_strut_num_joints = 3; +spool_strut_tab_width=8; +spool_strut_tab_width_narrow=6; +spool_strut_tab_outset=8; + +spool_width_slop = 1.4; // amount of slop for the spool assembly side-to-side inside the enclosure +spool_width = flap_width - flap_notch_depth*2 + flap_width_slop + thickness*2; // spool width, outside face (spool to spool) +spool_width_clearance = max(spool_width, flap_width + flap_width_slop); // width clearance for the spool, either for the spool itself or the flaps + +spool_strut_width = (spool_strut_tab_outset + thickness/2) * 2; +spool_strut_length = spool_width; +spool_strut_inner_length = spool_width - 3 * thickness; + + +spool_tab_clearance = -0.06; // for the tabs connecting the struts to the spool ends (interference fit) +spool_retaining_clearance = 0.10; // for the notches in the spool retaining wall +spool_joint_clearance = 0.10; // for the notched joints on the spool struts + +spool_strut_exclusion_radius = sqrt((spool_strut_tab_outset+thickness/2)*(spool_strut_tab_outset+thickness/2) + (spool_strut_tab_width/2)*(spool_strut_tab_width/2)); +flap_pitch_radius = flap_spool_pitch_radius(num_flaps, flap_hole_radius, flap_hole_separation); //num_flaps * (flap_hole_radius*2 + flap_hole_separation) / (2*PI); +echo(spool_strut_exclusion_radius ); +echo(flap_pitch_radius ); +echo((spool_strut_exclusion_radius + flap_pitch_radius)/2); +magnet_diameter = 4; +magnet_hole_clearance = -0.07; // interference fit +magnet_hole_radius = (magnet_diameter + magnet_hole_clearance)/2; + +magnet_hole_offset = (spool_strut_exclusion_radius + flap_pitch_radius)/2; + + + +spool_explosion = 0; // 0-1 +spool_strut_explosion = lookup(spool_explosion, [ + [0, 0], + [0.2, 0], + [1, 30], +]); +spool_horizontal_explosion = lookup(spool_explosion, [ + [0, 0], + [1, 8], +]); +flap_spool_outset = flap_hole_radius; + + +module spool_strut() { + joint_tab_width = spool_strut_inner_length / spool_strut_num_joints; + linear_extrude(thickness, center=true) { + union() { + translate([0, -spool_strut_tab_width_narrow / 2]) { + square([thickness + eps, spool_strut_tab_width_narrow]); + } + translate([thickness, -spool_strut_tab_width / 2]) { + square([spool_strut_length - thickness, spool_strut_tab_width]); + } + translate([thickness*2, -spool_strut_width / 2]) { + difference() { + square([spool_strut_inner_length, spool_strut_width]); + + // subtract out joints + union() { + for (i = [0:2:spool_strut_num_joints-1]) { + translate([i*joint_tab_width - spool_joint_clearance/2, -eps]) + square([joint_tab_width + spool_joint_clearance, thickness + spool_joint_clearance/2 + eps]); + } + for (i = [1:2:spool_strut_num_joints-1]) { + translate([i*joint_tab_width, spool_strut_width - thickness - spool_joint_clearance/2]) + square([joint_tab_width, thickness + spool_joint_clearance + eps]); + } + } + } + } + } + } +} + +module spool_struts() { + for (i=[0:3]) { + angle = 90*i; + //color([i < 2 ? 0 : 1, i == 0 || i == 2 ? 0 : 1, 0]) + color(i % 2 == 0 ? assembly_color2 : assembly_color3) + translate([0, sin(angle)*(spool_strut_tab_outset + spool_strut_explosion), cos(angle)*(spool_strut_tab_outset + spool_strut_explosion)]) + rotate([-angle, 0, 0]) + spool_strut(); + } +} + +// ##### Struts for bracing spool ##### +module spool_strut_tab_hole(narrow, clearance) { + square([thickness + clearance, narrow ? spool_strut_tab_width_narrow + clearance : spool_strut_tab_width + clearance], center=true); +} +module spool_strut_tab_holes(narrow=false, clearance=spool_tab_clearance) { + for (i=[0:3]) { + angle = 90*i; + translate([cos(angle)*spool_strut_tab_outset, sin(angle)*spool_strut_tab_outset]) + rotate([0,0,angle]) + spool_strut_tab_hole(narrow, clearance); + } +} + +module flap_spool_complete(captive_nut=false, motor_shaft=false, magnet_hole=false) { + linear_extrude(thickness, convexity=10) { // 'convexity' to fix rendering errors with etch 'difference()' result + difference() { + flap_spool(num_flaps, flap_hole_radius, flap_hole_separation, flap_spool_outset, + height=0); + + spool_strut_tab_holes(narrow=captive_nut); + if (captive_nut) { + circle(r=m4_nut_width_corners_padded/2, $fn=6); + } + if (motor_shaft) { + rotate([0, 0, 90]) { + motor_shaft(); + } + } + if (magnet_hole) { + // Hole for press fit magnet + translate([magnet_hole_offset, 0]) { + circle(r=magnet_hole_radius, $fn=15); + } + } + } + } +} + +module flap_spool_etch() { + enclosure_etch_style() + flap_spool_home_indicator(num_flaps, flap_hole_radius, flap_hole_separation, flap_spool_outset); +} + + + +module flap_spool(flaps, flap_hole_radius, flap_hole_separation, outset, height) { + pitch_radius = flap_spool_pitch_radius(flaps, flap_hole_radius, flap_hole_separation); + outer_radius = flap_spool_outer_radius(flaps, flap_hole_radius, flap_hole_separation, outset); + + + module flap_spool_2d() { + difference() { + circle(r=outer_radius, $fn=60); + for (i = [0 : flaps - 1]) { + translate([cos(360/flaps*i)*pitch_radius, sin(360/flaps*i)*pitch_radius]) circle(r=flap_hole_radius, $fn=15); + } + } + } + + if (height > 0) { + linear_extrude(height) { + flap_spool_2d(); + } + } else { + flap_spool_2d(); + } +} + +module flap_spool_home_indicator(flaps, flap_hole_radius, flap_hole_separation, outset, height=0) { + pitch_radius = flap_spool_pitch_radius(flaps, flap_hole_radius, flap_hole_separation); + outer_radius = flap_spool_outer_radius(flaps, flap_hole_radius, flap_hole_separation, outset); + + module flap_spool_home_indicator_2d() { + angle = (360 / flaps) * round(0.25 * flaps); // always point directly to a flap hole + + translate([cos(angle) * pitch_radius, sin(angle) * pitch_radius]) + rotate([0, 0, angle]) + translate([-flap_hole_radius * 2, 0]) + hull() { + circle(r=flap_hole_radius/2, $fn=30); + translate([-flap_hole_radius * 1.25, 0]) + circle(r=flap_hole_radius/2, $fn=30); + } + } + + if (height > 0) { + // convexity parameter fixes 'difference()' face polarity in preview mode... somehow + linear_extrude(height, convexity=2) + flap_spool_home_indicator_2d(); + } else { + flap_spool_home_indicator_2d(); + } +} + +function flap_spool_pitch_radius(flaps, flap_hole_radius, separation) = + flaps * (flap_hole_radius*2 + separation) / (2*PI); + +function flap_spool_outer_radius(flaps, flap_hole_radius, separation, outset) = + flap_spool_pitch_radius(flaps, flap_hole_radius, separation) + flap_hole_radius + outset; + +module spool_retaining_wall(m4_bolt_hole=false) { + linear_extrude(thickness) { + difference() { + square([spool_strut_width, spool_strut_width], center=true); + spool_strut_tab_holes(clearance=spool_retaining_clearance); + if (m4_bolt_hole) { + circle(r=m4_hole_diameter/2, $fn=30); + } + } + } +} diff --git a/3d/pcb.scad b/3d/pcb.scad index 7f8f9cdf..04efbd75 100644 --- a/3d/pcb.scad +++ b/3d/pcb.scad @@ -13,7 +13,7 @@ See the License for the specific language governing permissions and limitations under the License. */ -include ; +include ; pcb_thickness = 1.6; sensor_spool_distance = 0.70; // distance from the sensor to the face of the spool diff --git a/3d/splitflap.scad b/3d/splitflap.scad index 1b6916cb..d211eb1f 100644 --- a/3d/splitflap.scad +++ b/3d/splitflap.scad @@ -23,10 +23,16 @@ use; use; include<28byj-48.scad>; -include; include; -include; include; +include; +include; +include; +include; +include; +include; +include; +include; // ##### RENDERING OPTIONS ##### @@ -58,176 +64,21 @@ panel_horizontal = 0; render_revision = "deadbeef"; render_date = "YYYY-MM-DD"; -spool_explosion = 0; // 0-1 -spool_strut_explosion = lookup(spool_explosion, [ - [0, 0], - [0.2, 0], - [1, 30], -]); -spool_horizontal_explosion = lookup(spool_explosion, [ - [0, 0], - [1, 8], -]); - - // Ponoko kerf values are 0.2 mm for MDF and acrylic (all thicknesses) // Remember: it's better to underestimate (looser fit) than overestimate (no fit) kerf_width = 0.2 - 0.02; -// MDF, .120in nominal -// https://www.ponoko.com/materials/mdf-fiberboard -thickness = 3.0; - -etch_depth = 0.1; // for render -eps=.01; - -captive_nut_inset=6; - -assembly_inner_radius = m4_hole_diameter/2; - - -// Rendering Colors -assembly_color = [0.76, 0.60, 0.42]; // MDF, "c1996b" -etch_color = [0, 0, 0]; // black, "000000" - -hardware_color = [0.75, 0.75, 0.8]; // steel, "bfbfcc" - -flap_color = [1, 1, 1]; // white, "ffffff" - - -// multiply two equal matricies by each element, limiting to a max of 1.0 -function color_multiply(x, y) = - [ for(j=[0:len(x) - 1]) min(x[j] * y[j], 1.0) ]; - -// inverts a color matrix by subtracting the input channel values from 1.0 -function color_invert(x) = - [ for(j=[0:len(x) - 1]) (1.0 - x[j]) ]; - -assembly_color1 = color_multiply(assembly_color, [1.161, 1.157, 1.157, 1.0]); // "e1b17c" with MDF -assembly_color2 = color_multiply(assembly_color, [0.897, 0.895, 0.895, 1.0]); // "ae8960" with MDF -assembly_color3 = color_multiply(assembly_color, [0.547, 0.542, 0.540, 1.0]); // "6a533a" with MDF -assembly_color4 = color_multiply(assembly_color, [0.268, 0.268, 0.271, 1.0]); // "34291d" with MDF - -bolt_color = hardware_color; -nut_color = color_multiply(hardware_color, [0.933, 0.933, 0.9, 1.0]); // "b2b2b7" with steel - -letter_color = color_invert(flap_color); // inverse of the flap color, for contrast - - -flap_rendered_angle = 90; - - -flap_width_slop = 0.5; // amount of slop of the flap side to side between the 2 spools - -spool_width_slop = 1.4; // amount of slop for the spool assembly side-to-side inside the enclosure - -spool_tab_clearance = -0.06; // for the tabs connecting the struts to the spool ends (interference fit) -spool_retaining_clearance = 0.10; // for the notches in the spool retaining wall -spool_joint_clearance = 0.10; // for the notched joints on the spool struts - - num_flaps = 40; -flap_hole_radius = (flap_pin_width + 1) / 2; -flap_hole_separation = 1; // additional spacing between hole edges -flap_gap = (flap_hole_radius * 2 - flap_pin_width) + flap_hole_separation; function get_flap_gap() = flap_gap; // for exposing this value when this file is 'used' and not 'included' in other files -flap_spool_outset = flap_hole_radius; -flap_pitch_radius = flap_spool_pitch_radius(num_flaps, flap_hole_radius, flap_hole_separation); //num_flaps * (flap_hole_radius*2 + flap_hole_separation) / (2*PI); spool_outer_radius = flap_spool_outer_radius(num_flaps, flap_hole_radius, flap_hole_separation, flap_spool_outset); //flap_pitch_radius + 2*flap_hole_radius; -// Radius where flaps are expected to flap in their *most collapsed* (90 degree) state -exclusion_radius = sqrt(flap_height*flap_height + flap_pitch_radius*flap_pitch_radius); -// Radius where flaps are expected to flap in their *most extended* state -outer_exclusion_radius = flap_pitch_radius + flap_height + 2; - -front_forward_offset = flap_pitch_radius + flap_thickness/2; - -flap_notch_height = (flap_notch_height_auto == true) ? sqrt(spool_outer_radius*spool_outer_radius - flap_pitch_radius*flap_pitch_radius) : flap_notch_height_default; - -spool_width = flap_width - flap_notch_depth*2 + flap_width_slop + thickness*2; // spool width, outside face (spool to spool) -spool_width_clearance = max(spool_width, flap_width + flap_width_slop); // width clearance for the spool, either for the spool itself or the flaps - //legacyAssert(spool_width >= flap_width, "Flap is wider than spool!"); -spool_strut_num_joints = 3; -spool_strut_tab_width=8; -spool_strut_tab_width_narrow=6; -spool_strut_tab_outset=8; -spool_strut_width = (spool_strut_tab_outset + thickness/2) * 2; -spool_strut_length = spool_width; -spool_strut_inner_length = spool_width - 3 * thickness; - -spool_strut_exclusion_radius = sqrt((spool_strut_tab_outset+thickness/2)*(spool_strut_tab_outset+thickness/2) + (spool_strut_tab_width/2)*(spool_strut_tab_width/2)); - - -magnet_diameter = 4; -magnet_hole_clearance = -0.07; // interference fit -magnet_hole_radius = (magnet_diameter + magnet_hole_clearance)/2; -magnet_hole_offset = (spool_strut_exclusion_radius + flap_pitch_radius)/2; - -// Clearance between the motor chassis and the outside right wall of the previous module -28byj48_chassis_height_clearance = 1.4; - -motor_shaft_under_radius = 0.08; // interference fit -motor_slop_radius = 3; - - -// Width measured from the outside of the walls -enclosure_wall_to_wall_width = thickness + spool_width_slop/2 + spool_width_clearance + spool_width_slop/2 + max(28byj48_mount_bracket_height + m4_button_head_length, 4 + 28byj48_mount_bracket_height - spool_width_slop/2) + thickness; - -// Width of the front panel -enclosure_width = enclosure_wall_to_wall_width + 28byj48_chassis_height + 28byj48_chassis_height_clearance - thickness - 28byj48_mount_bracket_height; -enclosure_horizontal_inset = (enclosure_width - enclosure_wall_to_wall_width)/2; -front_window_upper_base = (flap_height - flap_pin_width/2); -front_window_overhang = 3; -front_window_upper = front_window_upper_base - front_window_overhang; -front_window_lower = sqrt(outer_exclusion_radius*outer_exclusion_radius - front_forward_offset*front_forward_offset); -front_window_slop = 0; -front_window_width = spool_width_slop + spool_width_clearance + front_window_slop; -front_window_right_inset = thickness - front_window_slop/2; -enclosure_vertical_margin = 10; // gap between top/bottom of flaps and top/bottom of enclosure -enclosure_vertical_inset = max(thickness*1.5, m4_nut_width_corners_padded/2); // distance from top of sides to top of the top piece -enclosure_height_upper = exclusion_radius + enclosure_vertical_margin + thickness + enclosure_vertical_inset; -enclosure_height_lower = flap_pitch_radius + flap_height + enclosure_vertical_margin + thickness + enclosure_vertical_inset; -enclosure_height = enclosure_height_upper + enclosure_height_lower; - -enclosure_horizontal_rear_margin = thickness; // minumum distance between the farthest feature and the rear - -enclosure_length = front_forward_offset + 28byj48_mount_center_offset + m4_hole_diameter/2 + enclosure_horizontal_rear_margin; // distance from the outside spool face to the inside of the left enclosure pcb_to_spool = enclosure_wall_to_wall_width - front_window_width - thickness + spool_width_slop/2; - -// Enclosure tabs: front/back -enclosure_tab_clearance = 0.10; - -num_front_tabs = 2; -front_tab_width = (enclosure_wall_to_wall_width - 2*thickness) / (num_front_tabs*2 - 1); - -enclosure_length_right = front_forward_offset + m4_hole_diameter/2 + 2; - -side_tab_width = enclosure_length_right / 4; -side_tab_width_fraction = 0.5; - - -backstop_bolt_vertical_offset = - (exclusion_radius + outer_exclusion_radius)/2; -backstop_bolt_forward_range = 14; - -motor_mount_hole_radius = m4_hole_diameter/2; -motor_backpack_extent = 28byj48_backpack_extent + 2; // Add 2mm to make sure there's room for the wires -motor_hole_slop = 1; -motor_window_radius = 5; - -connector_bracket_length_outer = 14; -connector_bracket_length_inner = side_tab_width * 2 - m4_button_head_diameter/2; -connector_bracket_thickness = captive_nut_inset - thickness - 0.2; -connector_bracket_width = enclosure_width - enclosure_wall_to_wall_width + thickness*2 + connector_bracket_thickness*2; -connector_bracket_overlap = 4; -connector_bracket_clearance = 0.40; -connector_bracket_depth_clearance = 0.20; - // 'get' functions to extract these values for when this file is 'used' and not 'included' function connector_bracket_length() = connector_bracket_length_outer; function connector_bracket_width() = connector_bracket_width; @@ -278,446 +129,8 @@ echo(flap_hole_radius=flap_hole_radius); echo(flap_notch_height=flap_notch_height); echo(pcb_to_sensor=pcb_to_sensor(pcb_to_spool)); - -module standard_m4_bolt(nut_distance=-1) { - if (render_bolts) { - color(bolt_color) - roughM4_7380(10); - if (nut_distance >= 0) { - color(nut_color) { - translate([0, 0, nut_distance]) { - linear_extrude(m4_nut_length) { - difference() { - circle(r=m4_nut_width_corners/2, $fn=6); - circle(r=m4_hole_diameter/2, $fn=20); - } - } - } - } - } - } -} - - -// ##### CAPTIVE NUT NEGATIVE ##### - -// Centered in the x dimension -module captive_nut(bolt_diameter, bolt_length, nut_width, nut_length, nut_inset) { - union() { - translate([-bolt_diameter/2, 0, 0]) - square([bolt_diameter, bolt_length]); - translate([-nut_width/2, nut_inset, 0]) - square([nut_width, nut_length]); - } -} -module m4_captive_nut(bolt_length=m4_bolt_length) { - captive_nut(m4_hole_diameter, bolt_length + 1, m4_nut_width_flats, m4_nut_length_padded, captive_nut_inset); -} - - -module zip_tie_holes() { - spacing = (zip_tie_spacing + zip_tie_width)/2; - - translate([-spacing, 0, 0]) - rounded_square([zip_tie_width, zip_tie_height], center=true, r=zip_tie_fillet, $fn=30); - translate([spacing, 0, 0]) - rounded_square([zip_tie_width, zip_tie_height], center=true, r=zip_tie_fillet, $fn=30); -} - - -// ##### Struts for bracing spool ##### -module spool_strut_tab_hole(narrow, clearance) { - square([thickness + clearance, narrow ? spool_strut_tab_width_narrow + clearance : spool_strut_tab_width + clearance], center=true); -} -module spool_strut_tab_holes(narrow=false, clearance=spool_tab_clearance) { - for (i=[0:3]) { - angle = 90*i; - translate([cos(angle)*spool_strut_tab_outset, sin(angle)*spool_strut_tab_outset]) - rotate([0,0,angle]) - spool_strut_tab_hole(narrow, clearance); - } -} -module spool_strut() { - joint_tab_width = spool_strut_inner_length / spool_strut_num_joints; - linear_extrude(thickness, center=true) { - union() { - translate([0, -spool_strut_tab_width_narrow / 2]) { - square([thickness + eps, spool_strut_tab_width_narrow]); - } - translate([thickness, -spool_strut_tab_width / 2]) { - square([spool_strut_length - thickness, spool_strut_tab_width]); - } - translate([thickness*2, -spool_strut_width / 2]) { - difference() { - square([spool_strut_inner_length, spool_strut_width]); - - // subtract out joints - union() { - for (i = [0:2:spool_strut_num_joints-1]) { - translate([i*joint_tab_width - spool_joint_clearance/2, -eps]) - square([joint_tab_width + spool_joint_clearance, thickness + spool_joint_clearance/2 + eps]); - } - for (i = [1:2:spool_strut_num_joints-1]) { - translate([i*joint_tab_width, spool_strut_width - thickness - spool_joint_clearance/2]) - square([joint_tab_width, thickness + spool_joint_clearance + eps]); - } - } - } - } - } - } -} - -module spool_struts() { - for (i=[0:3]) { - angle = 90*i; - //color([i < 2 ? 0 : 1, i == 0 || i == 2 ? 0 : 1, 0]) - color(i % 2 == 0 ? assembly_color2 : assembly_color3) - translate([0, sin(angle)*(spool_strut_tab_outset + spool_strut_explosion), cos(angle)*(spool_strut_tab_outset + spool_strut_explosion)]) - rotate([-angle, 0, 0]) - spool_strut(); - } -} - - -module flap_spool_complete(captive_nut=false, motor_shaft=false, magnet_hole=false) { - linear_extrude(thickness, convexity=10) { // 'convexity' to fix rendering errors with etch 'difference()' result - difference() { - flap_spool(num_flaps, flap_hole_radius, flap_hole_separation, flap_spool_outset, - height=0); - - spool_strut_tab_holes(narrow=captive_nut); - if (captive_nut) { - circle(r=m4_nut_width_corners_padded/2, $fn=6); - } - if (motor_shaft) { - rotate([0, 0, 90]) { - motor_shaft(); - } - } - if (magnet_hole) { - // Hole for press fit magnet - translate([magnet_hole_offset, 0]) { - circle(r=magnet_hole_radius, $fn=15); - } - } - } - } -} - -module flap_spool_etch() { - enclosure_etch_style() - flap_spool_home_indicator(num_flaps, flap_hole_radius, flap_hole_separation, flap_spool_outset); -} - -module spool_retaining_wall(m4_bolt_hole=false) { - linear_extrude(thickness) { - difference() { - square([spool_strut_width, spool_strut_width], center=true); - spool_strut_tab_holes(clearance=spool_retaining_clearance); - if (m4_bolt_hole) { - circle(r=m4_hole_diameter/2, $fn=30); - } - } - } -} - - -module flap_2d(cut_tabs = true) { - translate([0, -flap_pin_width/2, 0]) - difference() { - union() { - square([flap_width, flap_height - flap_corner_radius]); - - // rounded corners - hull() { - translate([flap_corner_radius, flap_height - flap_corner_radius]) - circle(r=flap_corner_radius, $fn=40); - translate([flap_width - flap_corner_radius, flap_height - flap_corner_radius]) - circle(r=flap_corner_radius, $fn=40); - } - } - // spool tabs - if(cut_tabs) { - translate([-eps, flap_pin_width]) - square([eps + flap_notch_depth, flap_notch_height]); - translate([flap_width - flap_notch_depth, flap_pin_width]) - square([eps + flap_notch_depth, flap_notch_height]); - } - } -} - -module flap() { - color(flap_color) - translate([0, 0, -flap_thickness/2]) - linear_extrude(height=flap_thickness) { - flap_2d(); - } -} - -module draw_letter(letter) { - translate([0, -flap_height * letter_height, 0]) // valign compensation - scale([letter_width, 1, 1]) - translate([letter_offset_x, letter_offset_y]) - text(text=letter, size=flap_height * letter_height * 2, font=letter_font, halign="center"); -} - -module flap_letter(letter, half = 0) { - color(letter_color) - translate([0, 0, flap_thickness/2 + eps]) - linear_extrude(height=0.1, center=true) { - if (half != 0) { // trimming to top (1) or bottom (2) - intersection() { - flap_2d(); // limit to bounds of flap - translate([flap_width/2, -flap_pin_width/2, 0]) { - rotation = (half == 2) ? -180 : 0; // flip upside-down for bottom - gap_comp = (letter_gap_comp == true) ? -flap_gap/2 : 0; - translate([0, gap_comp, 0]) - rotate([0,0,rotation]) - draw_letter(letter); - } - } - } else { - translate([flap_width/2, -flap_pin_width/2 - flap_gap/2, 0]) - draw_letter(letter); - } - } -} - - -// double-flatted motor shaft of 28byj-48 motor (2D) -module motor_shaft() { - union() { - intersection() { - circle(r=28byj48_shaft_radius-motor_shaft_under_radius, $fn=50); - square([28byj48_shaft_radius*2, 3], center=true); - } - square([28byj48_shaft_radius/3, 28byj48_shaft_radius*4], center=true); - } -} - -module front_tabs_negative() { - for (i = [0 : num_front_tabs-1]) { - translate([thickness + (i*2+0.5) * front_tab_width, 0, 0]) - square([front_tab_width + enclosure_tab_clearance, thickness + enclosure_tab_clearance], center=true); - } - for (i = [0 : num_front_tabs-2]) { - translate([thickness + (i*2+1.5) * front_tab_width, 0, 0]) - circle(r=m4_hole_diameter/2, $fn=30); - } -} - -module connector_bracket_2d() { - difference() { - square([connector_bracket_width, connector_bracket_length_outer]); - translate([connector_bracket_thickness, -eps]) { - square([connector_bracket_width - connector_bracket_thickness*2, connector_bracket_length_outer - connector_bracket_length_inner + eps]); - } - translate([connector_bracket_thickness - connector_bracket_clearance/2, -eps]) { - square([thickness + connector_bracket_clearance, connector_bracket_length_outer - connector_bracket_overlap + connector_bracket_depth_clearance + eps]); - } - translate([connector_bracket_width - connector_bracket_thickness - thickness - connector_bracket_clearance/2, -eps]) { - square([thickness + connector_bracket_clearance, connector_bracket_length_outer - connector_bracket_overlap + connector_bracket_depth_clearance + eps]); - } - } -} - -module connector_bracket() { - linear_extrude(height=thickness) { - connector_bracket_2d(); - } -} - -module enclosure_etch_style() { - color(etch_color) - translate([0, 0, thickness]) - linear_extrude(height=etch_depth) - children(); -} - -module enclosure_front() { - linear_extrude(height=thickness) { - difference() { - translate([-enclosure_horizontal_inset, 0, 0]) { - square([enclosure_width, enclosure_height]); - } - - // Viewing window cutout - translate([front_window_right_inset, enclosure_height_lower - front_window_lower]) - square([front_window_width, front_window_lower + front_window_upper]); - - // Front lower tabs - translate([0, thickness * 0.5 + enclosure_vertical_inset, 0]) - front_tabs_negative(); - - // Front upper tabs - translate([0, enclosure_height - thickness * 0.5 - enclosure_vertical_inset, 0]) - front_tabs_negative(); - } - } -} - -module enclosure_front_etch() { - // alignment indicator, left side (triangle) - enclosure_etch_style() - translate([enclosure_wall_to_wall_width - thickness - enclosure_indicator_inset, enclosure_indicator_position_y]) - triangle(enclosure_indicator_size, center=true); - - // alignment indicator, right side (circle) - enclosure_etch_style() - translate([thickness + enclosure_indicator_inset, enclosure_indicator_position_y]) - circle(r=enclosure_indicator_size/2, $fn=60); - - // position indicator, 'up' arrow - enclosure_etch_style() - translate([enclosure_width - enclosure_horizontal_inset * 1.5, enclosure_height - enclosure_indicator_arrow_height/2 - enclosure_indicator_inset]) - arrow([enclosure_indicator_arrow_width, enclosure_indicator_arrow_height], center=true); -} - -// holes for 28byj-48 motor, centered around motor shaft -module motor_mount() { - circle(r=28byj48_shaft_radius+motor_slop_radius, $fn=30); - translate([-28byj48_mount_center_offset, -8]) { - circle(r=motor_mount_hole_radius, $fn=30); - } - translate([28byj48_mount_center_offset, -8]) { - circle(r=motor_mount_hole_radius, $fn=30); - } - - hull() { - x = -28byj48_chassis_radius - motor_hole_slop/2 + motor_window_radius; - y = [-28byj48_shaft_offset - motor_backpack_extent - motor_hole_slop/2 + motor_window_radius, - -28byj48_shaft_offset + 28byj48_chassis_radius + motor_hole_slop/2 - motor_window_radius]; - - translate([ x, y[0], 0]) circle(r=motor_window_radius, $fn=40); - translate([-x, y[1], 0]) circle(r=motor_window_radius, $fn=40); - translate([-x, y[0], 0]) circle(r=motor_window_radius, $fn=40); - translate([ x, y[1], 0]) circle(r=motor_window_radius, $fn=40); - } -} - -module side_tabs_negative(hole_sizes=[], extend_last_tab=false) { - for (i = [0 : len(hole_sizes)]) { - length = (extend_last_tab && i == len(hole_sizes)) ? side_tab_width * side_tab_width_fraction + eps : side_tab_width * side_tab_width_fraction; - translate([0, thickness + (i*2) * side_tab_width + side_tab_width * (1 - side_tab_width_fraction)/2 + length/2, 0]) - square([thickness + enclosure_tab_clearance, length + enclosure_tab_clearance], center=true); - } - for (i = [0 : len(hole_sizes) - 1]) { - hole_size = hole_sizes[i]; - if (hole_size > 0) { - bolt_head_hole = hole_size == 2; - translate([0, thickness + (i*2 + 1.5) * side_tab_width, 0]) - circle(r=(bolt_head_hole ? m4_button_head_diameter : m4_hole_diameter)/2, $fn=30); - } - } -} - -module backstop_bolt_slot(radius) { - hull() { - circle(r=radius, $fn=15); - translate([0, backstop_bolt_forward_range]) { - circle(r=radius, $fn=15); - } - } -} - -module connector_bracket_side_holes() { - // overlap slot - translate([enclosure_vertical_inset - thickness - connector_bracket_clearance/2, -connector_bracket_overlap]) { - square([thickness + connector_bracket_clearance, connector_bracket_overlap + eps]); - } -} - -module alignment_bar() { - color(assembly_color1) - translate([enclosure_width - enclosure_horizontal_inset, -enclosure_length_right + front_forward_offset - alignment_bar_diameter/2, -enclosure_height_lower + alignment_bar_diameter/2]) - rotate([0, -90, 0]) - linear_extrude(height=enclosure_width * len(render_letters)) - circle(r=alignment_bar_diameter/2, $fn=60); -} - -module enclosure_left() { - linear_extrude(height=thickness) { - difference() { - square([enclosure_height, enclosure_length]); - translate([enclosure_height_lower, enclosure_length - front_forward_offset, 0]) - circle(r=m4_hole_diameter/2, $fn=30); - - translate([enclosure_height_lower, enclosure_length - front_forward_offset]) { - rotate([0, 0, 90]) { - motor_mount(); - } - } - - // bottom side tabs - translate([thickness * 0.5 + enclosure_vertical_inset, enclosure_length, 0]) - mirror([0, 1, 0]) - side_tabs_negative(hole_sizes=[1]); - - // top side tabs - translate([enclosure_height - thickness * 0.5 - enclosure_vertical_inset, enclosure_length, 0]) - mirror([0, 1, 0]) - side_tabs_negative(hole_sizes=[1]); - - // Connector bracket cuts - translate([enclosure_height, enclosure_length]) { - mirror([1, 0, 0]) { - connector_bracket_side_holes(); - } - } - translate([0, enclosure_length]) { - connector_bracket_side_holes(); - } - - - // PCB mounting holes - translate([enclosure_height_lower - magnet_hole_offset - pcb_hole_to_sensor_y, enclosure_length - front_forward_offset - pcb_hole_to_sensor_x]) { - rotate([180, 0, 0]) { - rotate([0, 0, -90]) { - pcb_cutouts(); - } - } - } - - // Zip tie holes, sensor (leading to bottom) - translate([enclosure_left_zip_bottom_inset, zip_tie_height/2 + enclosure_left_zip_side_inset, 0]) - zip_tie_holes(); - - // Zip tie holes, motor (leading to top) - translate([enclosure_height - enclosure_left_zip_top_inset, enclosure_length - front_forward_offset]) - rotate([0, 0, 90]) // cable channel facing 'up' - zip_tie_holes(); - - // Alignment bar cutout - translate([0, alignment_bar_center]) { - union() { - // Cutout - translate([alignment_bar_cutout_width/2, 0]) - circle(r=alignment_bar_cutout_width/2, $fn=40); - square([alignment_bar_cutout_width, alignment_bar_cutout_width], center=true); - - // Front-side fillet - // translate([0, alignment_bar_cutout_width/2, 0]) - // fillet_tool(r=alignment_bar_fillet_radius, overlap=1, $fn=40); - - // Back-side fillet - translate([0, -alignment_bar_cutout_width/2, 0]) - mirror([0, 1, 0]) - fillet_tool(r=alignment_bar_fillet_radius, $fn=40); - } - } - } - } -} - -module enclosure_left_etch() { - // alignment indicator (triangle) - enclosure_etch_style() - translate([enclosure_indicator_position_y, enclosure_length - enclosure_indicator_inset]) - rotate([0, 0, -90]) - triangle(enclosure_indicator_size, center=true); -} - +// Dead code? +assembly_inner_radius = m4_hole_diameter/2; module shaft_centered_motor_hole() { margin = 5; width = 28byj48_mount_center_offset*2 + 3.5*2 + margin*2; @@ -727,604 +140,11 @@ module shaft_centered_motor_hole() { square([width, length]); } -module enclosure_right() { - linear_extrude(height=thickness) { - difference() { - square([enclosure_height, enclosure_length_right]); - translate([enclosure_height_upper, enclosure_length_right - front_forward_offset, 0]) - circle(r=m4_hole_diameter/2, $fn=30); - - // backstop bolt slot - translate([enclosure_height_upper - backstop_bolt_vertical_offset, enclosure_length_right - front_forward_offset, 0]) { - backstop_bolt_slot(radius = m4_hole_diameter/2); - } - - // top side tabs - translate([0.5*thickness + enclosure_vertical_inset, enclosure_length_right, 0]) - mirror([0, 1, 0]) - side_tabs_negative(hole_sizes=[1]); - - // bottom side tabs - translate([enclosure_height - 0.5*thickness - enclosure_vertical_inset, enclosure_length_right, 0]) - mirror([0, 1, 0]) - side_tabs_negative(hole_sizes=[1]); - - // Connector bracket cuts - translate([enclosure_height, enclosure_length_right]) { - mirror([1, 0, 0]) { - connector_bracket_side_holes(); - } - } - translate([0, enclosure_length_right]) { - connector_bracket_side_holes(); - } - } - } -} - -module enclosure_right_etch() { - // alignment indicator (circle) - enclosure_etch_style() - translate([enclosure_height - enclosure_indicator_position_y, enclosure_length_right - enclosure_indicator_inset]) - circle(r=enclosure_indicator_size/2, $fn=60); -} - -module front_back_tabs() { - for (i = [0 : 2 : num_front_tabs*2-2]) { - translate([i * front_tab_width, -eps, 0]) - square([front_tab_width, thickness + eps]); - } -} - -module side_tabs(tabs) { - for (i = [0 : 2 : tabs*2-2]) { - translate([-eps, i * side_tab_width + side_tab_width * (1 - side_tab_width_fraction)/2, 0]) - square([thickness + eps, side_tab_width * side_tab_width_fraction]); - } -} - -module front_back_captive_nuts() { - for (i = [0 : 2 : num_front_tabs-1]) { - translate([(i*2 + 1.5) * front_tab_width, -thickness, 0]) - m4_captive_nut(); - } -} - -module side_captive_nuts(hole_types=[]) { - for (i = [0 : len(hole_types)-1]) { - hole_type = hole_types[i]; - translate([-thickness, (i*2 + 1.5) * side_tab_width, 0]) { - rotate([0, 0, -90]) { - if (hole_type == 2) { - } else if (hole_type == 1) { - m4_captive_nut(); - } - } - } - } -} - - -module enclosure_top() { - // note, this is flipped upside down (around the x axis) when assembled so the clean side faces out - linear_extrude(height = thickness) { - translate([thickness, 0, 0]) { - difference() { - union() { - square([enclosure_wall_to_wall_width - 2 * thickness, enclosure_length_right]); - - // front tabs - mirror([0, 1, 0]) - front_back_tabs(); - - // left tabs - translate([enclosure_wall_to_wall_width - 2 * thickness, thickness, 0]) - side_tabs(2); - - // right tabs - mirror([1, 0, 0]) - translate([0, thickness, 0]) - side_tabs(2); - } - - // front captive nuts - front_back_captive_nuts(); - - // right captive nuts - translate([0, thickness, 0]) - side_captive_nuts(hole_types = [1]); - - // left captive nuts - translate([enclosure_wall_to_wall_width - 2 * thickness, thickness, 0]) - mirror([1, 0, 0]) - side_captive_nuts(hole_types = [1]); - - // mounting hole - translate([(enclosure_wall_to_wall_width - 2 * thickness)/2, enclosure_length_right - mounting_hole_inset]) { - circle(r=m4_hole_diameter/2, $fn=30); - } - } - } - } -} - -module enclosure_bottom() { - linear_extrude(height = thickness) { - translate([thickness, 0, 0]) { - difference() { - union() { - square([enclosure_wall_to_wall_width - 2 * thickness, enclosure_length_right]); - - // front tabs - translate([0, enclosure_length_right, 0]) - front_back_tabs(); - - // left tabs - translate([enclosure_wall_to_wall_width - 2 * thickness, enclosure_length_right - thickness, 0]) - mirror([0, 1, 0]) - side_tabs(2); - - // right tabs - translate([0, enclosure_length_right - thickness, 0]) - mirror([0, 1, 0]) - mirror([1, 0, 0]) - side_tabs(2); - } - - // front captive nuts - translate([0, enclosure_length_right, 0]) - mirror([0,1,0]) - front_back_captive_nuts(); - - // right captive nuts - translate([0, enclosure_length_right - thickness, 0]) - mirror([0, 1, 0]) - side_captive_nuts(hole_types = [1]); - - // left captive nuts - translate([enclosure_wall_to_wall_width - 2 * thickness, enclosure_length_right - thickness, 0]) - mirror([0, 1, 0]) - mirror([1, 0, 0]) - side_captive_nuts(hole_types = [1]); - - // mounting hole - translate([(enclosure_wall_to_wall_width - 2 * thickness)/2, mounting_hole_inset]) { - circle(r=m4_hole_diameter/2, $fn=30); - } - } - } - } -} - -module enclosure_bottom_etch() { - enclosure_etch_style() - translate([captive_nut_inset + m4_nut_length + 1, 1, thickness]) { - text_label([str("rev. ", render_revision), render_date, "github.com/scottbez1/splitflap"]); - } -} - -module split_flap_3d(letter, include_connector) { - module position_front() { - translate([0, front_forward_offset + thickness, -enclosure_height_lower]) - rotate([90, 0, 0]) - children(); - } - - module positioned_front() { - position_front() - enclosure_front(); - } - - module positioned_front_etch() { - position_front() - enclosure_front_etch(); - } - - module position_left() { - translate([enclosure_wall_to_wall_width, -enclosure_length + front_forward_offset, -enclosure_height_lower]) - rotate([0, -90, 0]) - children(); - } - - module positioned_left() { - position_left() - enclosure_left(); - } - - module positioned_left_etch() { - position_left() - enclosure_left_etch(); - } - - module position_right() { - translate([0, -enclosure_length_right + front_forward_offset, enclosure_height_upper]) - rotate([0, 90, 0]) - children(); - } - - module positioned_right() { - position_right() - enclosure_right(); - } - - module positioned_right_etch() { - position_right() - enclosure_right_etch(); - } - - module positioned_top() { - translate([0, front_forward_offset, enclosure_height_upper - enclosure_vertical_inset]) - rotate([180, 0, 0]) - enclosure_top(); - } - - module position_bottom() { - translate([0, front_forward_offset - enclosure_length_right, -enclosure_height_lower + enclosure_vertical_inset]) - children(); - } - - module positioned_bottom() { - position_bottom() - enclosure_bottom(); - } - - module positioned_bottom_etch() { - position_bottom() - enclosure_bottom_etch(); - } - - module positioned_top_connector() { - translate([enclosure_wall_to_wall_width - thickness - connector_bracket_thickness, front_forward_offset - connector_bracket_length_outer, enclosure_height_upper - enclosure_vertical_inset]) { - connector_bracket(); - } - } - - module positioned_bottom_connector() { - translate([enclosure_wall_to_wall_width - thickness - connector_bracket_thickness, front_forward_offset - connector_bracket_length_outer, - enclosure_height_lower + enclosure_vertical_inset - thickness]) { - connector_bracket(); - } - } - - module positioned_left_bolts() { - // Top - translate([enclosure_wall_to_wall_width, front_forward_offset - (thickness + 1.5 * side_tab_width), enclosure_height_upper - enclosure_vertical_inset - thickness/2]) { - rotate([0, -90, 0]) { - standard_m4_bolt(nut_distance=captive_nut_inset); - } - } - - // Bottom - translate([enclosure_wall_to_wall_width, front_forward_offset - (thickness + 1.5 * side_tab_width), -enclosure_height_lower + enclosure_vertical_inset + thickness/2]) { - rotate([0, -90, 0]) { - standard_m4_bolt(nut_distance=captive_nut_inset); - } - } - } - - module positioned_right_bolts() { - // Top - translate([0, front_forward_offset - (thickness + 1.5 * side_tab_width), enclosure_height_upper - enclosure_vertical_inset - thickness/2]) { - rotate([0, 90, 0]) { - standard_m4_bolt(nut_distance=captive_nut_inset); - } - } - - // Bottom - translate([0, front_forward_offset - (thickness + 1.5 * side_tab_width), -enclosure_height_lower + enclosure_vertical_inset + thickness/2]) { - rotate([0, 90, 0]) { - standard_m4_bolt(nut_distance=captive_nut_inset); - } - } - } - - module positioned_front_bolts() { - // Top - translate([enclosure_wall_to_wall_width/2, front_forward_offset + thickness, enclosure_height_upper - enclosure_vertical_inset - thickness/2]) { - rotate([0, 90, -90]) { - standard_m4_bolt(nut_distance=captive_nut_inset); - } - } - - // Bottom - translate([enclosure_wall_to_wall_width/2, front_forward_offset + thickness, -enclosure_height_lower + enclosure_vertical_inset + thickness/2]) { - rotate([0, 90, -90]) { - standard_m4_bolt(nut_distance=captive_nut_inset); - } - } - } - - module positioned_enclosure() { - if (render_enclosure == 2) { - color(assembly_color1) - positioned_front(); - color(assembly_color2) - positioned_left(); - color(assembly_color2) - positioned_right(); - color(assembly_color3) - positioned_top(); - color(assembly_color3) - positioned_bottom(); - positioned_front_etch(); - positioned_left_etch(); - positioned_right_etch(); - positioned_bottom_etch(); - if (include_connector) { - color(assembly_color4) - positioned_top_connector(); - color(assembly_color4) - positioned_bottom_connector(); - } - if (render_bolts) { - positioned_left_bolts(); - positioned_right_bolts(); - positioned_front_bolts(); - } - } else if (render_enclosure == 1) { - %positioned_front(); - %positioned_left(); - %positioned_right(); - %positioned_top(); - %positioned_bottom(); - %positioned_front_etch(); - %positioned_left_etch(); - %positioned_right_etch(); - %positioned_bottom_etch(); - if (include_connector) { - %positioned_top_connector(); - %positioned_bottom_connector(); - } - if (render_bolts) { - positioned_left_bolts(); - positioned_right_bolts(); - positioned_front_bolts(); - } - } - } - - positioned_enclosure(); - if (render_pcb) { - translate([enclosure_wall_to_wall_width + eps, -pcb_hole_to_sensor_x, -magnet_hole_offset - pcb_hole_to_sensor_y]) { - rotate([0, 90, 0]) { - rotate([0, 0, 90]) { - pcb(pcb_to_spool, render_sensor_jig); - translate([0, 0, -thickness - 2 * eps]) { - standard_m4_bolt(nut_distance=thickness + pcb_thickness + 4*eps); - } - } - } - } - } - - translate([spool_width_slop/2 + thickness, 0, 0]) { - // Flap area - if (render_flaps > 0) { - rotate([0, 90, 0]) { - if (render_flap_area >= 1) { - translate([0, 0, thickness]) { - cylinder(r=exclusion_radius, h=flap_width - 2*thickness); - } - } - if (render_flap_area >= 2) { - translate([0, 0, thickness + (flap_width - 2*thickness)/4]) { - cylinder(r=outer_exclusion_radius, h=(flap_width - 2*thickness)/2); - } - } - } - - flap_offset = thickness > flap_notch_depth ? -flap_notch_depth + thickness + flap_width_slop/2 : flap_width_slop/2; - translate([flap_offset, 0, 0]) { - // Collapsed flaps on the top - for (i=[0:num_flaps/2 - 1]) { - if (i == 0 || render_flaps == 2) { - rotate([360/num_flaps * i, 0, 0]) { - translate([flap_width, flap_pitch_radius, 0]) { - rotate([flap_rendered_angle, 0, 180]) { - flap(); - if (i == 0) { - flap_letter(letter, 1); // 1 = top - } - } - } - } - } - } - - // Hanging flaps on the bottom - for (i=[1:num_flaps/2]) { - angle = -360/num_flaps*i; - translate([0, flap_pitch_radius*cos(angle), flap_pitch_radius * sin(angle)]) { - if (i == 1 || render_flaps == 2) { - rotate([-90, 0, 0]) { - flap(); - if (i == 1) { - flap_letter(letter, 2); // 2 = bottom - } - } - } - } - } - } - } - } - - if(render_spool) { - translate([(spool_width_clearance - spool_width + spool_width_slop) / 2 + thickness, 0, 0]) { - spool_struts(); - - // motor spool - translate([spool_width - thickness + 5*spool_horizontal_explosion, 0, 0]) { - rotate([0, 90, 0]) { - color(assembly_color) - flap_spool_complete(motor_shaft=true, magnet_hole=true); - translate([0, 0, -eps - thickness]) - flap_spool_etch(); - } - } - color(assembly_color1) { - translate([thickness - 3*spool_horizontal_explosion, 0, 0]) { - rotate([0, 90, 0]) { - spool_retaining_wall(m4_bolt_hole=true); - } - } - } - translate([-5*spool_horizontal_explosion, 0, 0]) { - rotate([0, 90, 0]) { - color(assembly_color) - flap_spool_complete(captive_nut=true); - flap_spool_etch(); - } - } - translate([thickness * 2, 0, 0]) { - rotate([0, -90, 0]) { - standard_m4_bolt(nut_distance=thickness + 7*spool_horizontal_explosion); - } - } - } - } - - if (render_motor) { - translate([enclosure_wall_to_wall_width - thickness - 28byj48_mount_bracket_height, 0, 0]) { - - rotate([-90, 0, 0]) { - - rotate([0, -90, 0]) { - Stepper28BYJ48(); - } - translate([0, -28byj48_shaft_offset, 0]) { - translate([0, 0, -28byj48_mount_center_offset]) { - rotate([0, 90, 0]) { - rotate([0, 0, 90]) { - standard_m4_bolt(nut_distance=thickness+28byj48_mount_bracket_height); - } - } - } - translate([0, 0, 28byj48_mount_center_offset]) { - rotate([0, 90, 0]) { - rotate([0, 0, 90]) { - standard_m4_bolt(nut_distance=thickness+28byj48_mount_bracket_height); - } - } - } - } - } - } - } -} - -module laser_etch() { - if (!render_2d_mirror && (render_etch || render_index == -1)) { - children(); - } -} - -module laser_mirror() { - if (render_2d_mirror) { - mirror([1, 0, 0]) - children(); - } - else { - children(); - } -} - if (render_3d) { for (i = [0 : render_units - 1]) { translate([-enclosure_width/2 + (-(render_units-1) / 2 + i)*(enclosure_width + render_unit_separation), 0, 0]) split_flap_3d(render_letters[render_units - 1 - i], include_connector=(i != render_units - 1)); } } else { - laser_mirror() { - panel_height = enclosure_length + kerf_width + enclosure_length_right + kerf_width + enclosure_width + kerf_width + spool_strut_width + kerf_width; - projection_renderer(render_index=render_index, render_etch=render_etch, kerf_width=kerf_width, panel_height=panel_height, panel_horizontal=panel_horizontal, panel_vertical=panel_vertical) { - // Main enclosure (left, right, front) - translate([0, 0]) - enclosure_left(); - - laser_etch() - translate([0, 0]) - enclosure_left_etch(); - - translate([0, enclosure_length + kerf_width]) - enclosure_right(); - - laser_etch() - translate([0, enclosure_length + kerf_width]) - enclosure_right_etch(); - - translate([0, enclosure_length + kerf_width + enclosure_length_right + kerf_width + enclosure_width - enclosure_horizontal_inset]) - rotate([0, 0, -90]) - enclosure_front(); - - laser_etch() - translate([0, enclosure_length + kerf_width + enclosure_length_right + kerf_width + enclosure_width - enclosure_horizontal_inset]) - rotate([0, 0, -90]) - enclosure_front_etch(); - - // Top and bottom - translate([enclosure_height + kerf_width + enclosure_length_right, enclosure_wall_to_wall_width + kerf_width]) - rotate([0, 0, 90]) - enclosure_top(); - - translate([enclosure_height + kerf_width, enclosure_wall_to_wall_width]) - rotate([0, 0, -90]) - enclosure_bottom(); - - // Bottom laser etching - laser_etch() - translate([enclosure_height + kerf_width, enclosure_wall_to_wall_width]) - rotate([0, 0, -90]) - enclosure_bottom_etch(); - - // Spool struts cut out of right side - translate([thickness*2 + 5, enclosure_length + kerf_width + enclosure_length_right - spool_strut_width/2 - 3, thickness]) - spool_strut(); - - // Spool struts at the top - spool_strut_y_offset = enclosure_length + kerf_width + enclosure_length_right + kerf_width + enclosure_width + kerf_width + spool_strut_width/2; - translate([spool_strut_length, spool_strut_y_offset, thickness/2]) - rotate([0, 0, 180]) - spool_strut(); - translate([spool_strut_length*2 + kerf_width, spool_strut_y_offset, thickness/2]) - rotate([0, 0, 180]) - spool_strut(); - translate([spool_strut_length*3 + kerf_width*2, spool_strut_y_offset, thickness/2]) - rotate([0, 0, 180]) - spool_strut(); - - // Connector brackets on the top right - translate([enclosure_height + kerf_width, 2 * enclosure_wall_to_wall_width + 2 * kerf_width - thickness, 0]) - connector_bracket(); - - translate([enclosure_height + kerf_width + connector_bracket_width - connector_bracket_length_outer, 2 * enclosure_wall_to_wall_width + 3 * kerf_width - thickness + connector_bracket_width + connector_bracket_length_outer, 0]) - rotate([0,0,-90]) - connector_bracket(); - - // Flap spools in flap window - flap_spool_y_off = enclosure_length + kerf_width + enclosure_length_right + kerf_width + enclosure_width - front_window_right_inset - enclosure_horizontal_inset - front_window_width/2; - flap_spool_x_off = spool_outer_radius + enclosure_height_lower - front_window_lower + kerf_width + 2; - translate([flap_spool_x_off, flap_spool_y_off]) - flap_spool_complete(motor_shaft=true, magnet_hole=true); - translate([flap_spool_x_off + spool_outer_radius*2 + 2, flap_spool_y_off]) - flap_spool_complete(captive_nut=true); - - // Flap spool etching - laser_etch() { - translate([flap_spool_x_off, flap_spool_y_off]) - mirror([0, 1, 0]) - flap_spool_etch(); - translate([flap_spool_x_off + spool_outer_radius*2 + 2, flap_spool_y_off]) - flap_spool_etch(); - } - - // Spool retaining wall in motor window - translate([enclosure_height_lower + 28byj48_shaft_offset - 28byj48_chassis_radius + (28byj48_chassis_radius + motor_backpack_extent)/2, enclosure_length - front_forward_offset - 28byj48_chassis_radius - motor_hole_slop/2 + spool_strut_width/2 + kerf_width]) - spool_retaining_wall(m4_bolt_hole=true); - - // Sensor soldering jig - translate([enclosure_height_lower + 28byj48_shaft_offset - 28byj48_chassis_radius + (28byj48_chassis_radius + motor_backpack_extent)/2 + sensor_jig_width(pcb_to_spool)/2, enclosure_length - front_forward_offset + 28byj48_chassis_radius + motor_hole_slop/2 - kerf_width]) - rotate([0, 0, 180]) - sensor_jig(pcb_to_spool); - } - } + split_flap_2d(); } diff --git a/3d/splitflap_2d.scad b/3d/splitflap_2d.scad new file mode 100644 index 00000000..92ed034b --- /dev/null +++ b/3d/splitflap_2d.scad @@ -0,0 +1,127 @@ +/* + Copyright 2015-2020 Scott Bezek and the splitflap contributors + + Licensed under the Apache License, Version 2.0 (the "License"); + you may not use this file except in compliance with the License. + You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + + Unless required by applicable law or agreed to in writing, software + distributed under the License is distributed on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + See the License for the specific language governing permissions and + limitations under the License. +*/ + +module laser_mirror() { + if (render_2d_mirror) { + mirror([1, 0, 0]) + children(); + } + else { + children(); + } +} + +module laser_etch() { + if (!render_2d_mirror && (render_etch || render_index == -1)) { + children(); + } +} + +module split_flap_2d() { + laser_mirror() { + panel_height = enclosure_length + kerf_width + enclosure_length_right + kerf_width + enclosure_width + kerf_width + spool_strut_width + kerf_width; + projection_renderer(render_index=render_index, render_etch=render_etch, kerf_width=kerf_width, panel_height=panel_height, panel_horizontal=panel_horizontal, panel_vertical=panel_vertical) { + // Main enclosure (left, right, front) + translate([0, 0]) + enclosure_left(); + + laser_etch() + translate([0, 0]) + enclosure_left_etch(); + + translate([0, enclosure_length + kerf_width]) + enclosure_right(); + + laser_etch() + translate([0, enclosure_length + kerf_width]) + enclosure_right_etch(); + + translate([0, enclosure_length + kerf_width + enclosure_length_right + kerf_width + enclosure_width - enclosure_horizontal_inset]) + rotate([0, 0, -90]) + enclosure_front(); + + laser_etch() + translate([0, enclosure_length + kerf_width + enclosure_length_right + kerf_width + enclosure_width - enclosure_horizontal_inset]) + rotate([0, 0, -90]) + enclosure_front_etch(); + + // Top and bottom + translate([enclosure_height + kerf_width + enclosure_length_right, enclosure_wall_to_wall_width + kerf_width]) + rotate([0, 0, 90]) + enclosure_top(); + + translate([enclosure_height + kerf_width, enclosure_wall_to_wall_width]) + rotate([0, 0, -90]) + enclosure_bottom(); + + // Bottom laser etching + laser_etch() + translate([enclosure_height + kerf_width, enclosure_wall_to_wall_width]) + rotate([0, 0, -90]) + enclosure_bottom_etch(); + + // Spool struts cut out of right side + translate([thickness*2 + 5, enclosure_length + kerf_width + enclosure_length_right - spool_strut_width/2 - 3, thickness]) + spool_strut(); + + // Spool struts at the top + spool_strut_y_offset = enclosure_length + kerf_width + enclosure_length_right + kerf_width + enclosure_width + kerf_width + spool_strut_width/2; + translate([spool_strut_length, spool_strut_y_offset, thickness/2]) + rotate([0, 0, 180]) + spool_strut(); + translate([spool_strut_length*2 + kerf_width, spool_strut_y_offset, thickness/2]) + rotate([0, 0, 180]) + spool_strut(); + translate([spool_strut_length*3 + kerf_width*2, spool_strut_y_offset, thickness/2]) + rotate([0, 0, 180]) + spool_strut(); + + // Connector brackets on the top right + translate([enclosure_height + kerf_width, 2 * enclosure_wall_to_wall_width + 2 * kerf_width - thickness, 0]) + connector_bracket(); + + translate([enclosure_height + kerf_width + connector_bracket_width - connector_bracket_length_outer, 2 * enclosure_wall_to_wall_width + 3 * kerf_width - thickness + connector_bracket_width + connector_bracket_length_outer, 0]) + rotate([0,0,-90]) + connector_bracket(); + + // Flap spools in flap window + flap_spool_y_off = enclosure_length + kerf_width + enclosure_length_right + kerf_width + enclosure_width - front_window_right_inset - enclosure_horizontal_inset - front_window_width/2; + flap_spool_x_off = spool_outer_radius + enclosure_height_lower - front_window_lower + kerf_width + 2; + translate([flap_spool_x_off, flap_spool_y_off]) + flap_spool_complete(motor_shaft=true, magnet_hole=true); + translate([flap_spool_x_off + spool_outer_radius*2 + 2, flap_spool_y_off]) + flap_spool_complete(captive_nut=true); + + // Flap spool etching + laser_etch() { + translate([flap_spool_x_off, flap_spool_y_off]) + mirror([0, 1, 0]) + flap_spool_etch(); + translate([flap_spool_x_off + spool_outer_radius*2 + 2, flap_spool_y_off]) + flap_spool_etch(); + } + + // Spool retaining wall in motor window + translate([enclosure_height_lower + 28byj48_shaft_offset - 28byj48_chassis_radius + (28byj48_chassis_radius + motor_backpack_extent)/2, enclosure_length - front_forward_offset - 28byj48_chassis_radius - motor_hole_slop/2 + spool_strut_width/2 + kerf_width]) + spool_retaining_wall(m4_bolt_hole=true); + + // Sensor soldering jig + translate([enclosure_height_lower + 28byj48_shaft_offset - 28byj48_chassis_radius + (28byj48_chassis_radius + motor_backpack_extent)/2 + sensor_jig_width(pcb_to_spool)/2, enclosure_length - front_forward_offset + 28byj48_chassis_radius + motor_hole_slop/2 - kerf_width]) + rotate([0, 0, 180]) + sensor_jig(pcb_to_spool); + } + } +} \ No newline at end of file diff --git a/3d/splitflap_3d.scad b/3d/splitflap_3d.scad new file mode 100644 index 00000000..9a7993ad --- /dev/null +++ b/3d/splitflap_3d.scad @@ -0,0 +1,326 @@ +/* + Copyright 2015-2020 Scott Bezek and the splitflap contributors + + Licensed under the Apache License, Version 2.0 (the "License"); + you may not use this file except in compliance with the License. + You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + + Unless required by applicable law or agreed to in writing, software + distributed under the License is distributed on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + See the License for the specific language governing permissions and + limitations under the License. +*/ + +flap_rendered_angle = 90; + +module split_flap_3d(letter, include_connector) { + module position_front() { + translate([0, front_forward_offset + thickness, -enclosure_height_lower]) + rotate([90, 0, 0]) + children(); + } + + module positioned_front() { + position_front() + enclosure_front(); + } + + module positioned_front_etch() { + position_front() + enclosure_front_etch(); + } + + module position_left() { + translate([enclosure_wall_to_wall_width, -enclosure_length + front_forward_offset, -enclosure_height_lower]) + rotate([0, -90, 0]) + children(); + } + + module positioned_left() { + position_left() + enclosure_left(); + } + + module positioned_left_etch() { + position_left() + enclosure_left_etch(); + } + + module position_right() { + translate([0, -enclosure_length_right + front_forward_offset, enclosure_height_upper]) + rotate([0, 90, 0]) + children(); + } + + module positioned_right() { + position_right() + enclosure_right(); + } + + module positioned_right_etch() { + position_right() + enclosure_right_etch(); + } + + module positioned_top() { + translate([0, front_forward_offset, enclosure_height_upper - enclosure_vertical_inset]) + rotate([180, 0, 0]) + enclosure_top(); + } + + module position_bottom() { + translate([0, front_forward_offset - enclosure_length_right, -enclosure_height_lower + enclosure_vertical_inset]) + children(); + } + + module positioned_bottom() { + position_bottom() + enclosure_bottom(); + } + + module positioned_bottom_etch() { + position_bottom() + enclosure_bottom_etch(); + } + + module positioned_top_connector() { + translate([enclosure_wall_to_wall_width - thickness - connector_bracket_thickness, front_forward_offset - connector_bracket_length_outer, enclosure_height_upper - enclosure_vertical_inset]) { + connector_bracket(); + } + } + + module positioned_bottom_connector() { + translate([enclosure_wall_to_wall_width - thickness - connector_bracket_thickness, front_forward_offset - connector_bracket_length_outer, - enclosure_height_lower + enclosure_vertical_inset - thickness]) { + connector_bracket(); + } + } + + module positioned_left_bolts() { + // Top + translate([enclosure_wall_to_wall_width, front_forward_offset - (thickness + 1.5 * side_tab_width), enclosure_height_upper - enclosure_vertical_inset - thickness/2]) { + rotate([0, -90, 0]) { + standard_m4_bolt(nut_distance=captive_nut_inset); + } + } + + // Bottom + translate([enclosure_wall_to_wall_width, front_forward_offset - (thickness + 1.5 * side_tab_width), -enclosure_height_lower + enclosure_vertical_inset + thickness/2]) { + rotate([0, -90, 0]) { + standard_m4_bolt(nut_distance=captive_nut_inset); + } + } + } + + module positioned_right_bolts() { + // Top + translate([0, front_forward_offset - (thickness + 1.5 * side_tab_width), enclosure_height_upper - enclosure_vertical_inset - thickness/2]) { + rotate([0, 90, 0]) { + standard_m4_bolt(nut_distance=captive_nut_inset); + } + } + + // Bottom + translate([0, front_forward_offset - (thickness + 1.5 * side_tab_width), -enclosure_height_lower + enclosure_vertical_inset + thickness/2]) { + rotate([0, 90, 0]) { + standard_m4_bolt(nut_distance=captive_nut_inset); + } + } + } + + module positioned_front_bolts() { + // Top + translate([enclosure_wall_to_wall_width/2, front_forward_offset + thickness, enclosure_height_upper - enclosure_vertical_inset - thickness/2]) { + rotate([0, 90, -90]) { + standard_m4_bolt(nut_distance=captive_nut_inset); + } + } + + // Bottom + translate([enclosure_wall_to_wall_width/2, front_forward_offset + thickness, -enclosure_height_lower + enclosure_vertical_inset + thickness/2]) { + rotate([0, 90, -90]) { + standard_m4_bolt(nut_distance=captive_nut_inset); + } + } + } + + module positioned_enclosure() { + if (render_enclosure == 2) { + color(assembly_color1) + positioned_front(); + color(assembly_color2) + positioned_left(); + color(assembly_color2) + positioned_right(); + color(assembly_color3) + positioned_top(); + color(assembly_color3) + positioned_bottom(); + positioned_front_etch(); + positioned_left_etch(); + positioned_right_etch(); + positioned_bottom_etch(); + if (include_connector) { + color(assembly_color4) + positioned_top_connector(); + color(assembly_color4) + positioned_bottom_connector(); + } + if (render_bolts) { + positioned_left_bolts(); + positioned_right_bolts(); + positioned_front_bolts(); + } + } else if (render_enclosure == 1) { + %positioned_front(); + %positioned_left(); + %positioned_right(); + %positioned_top(); + %positioned_bottom(); + %positioned_front_etch(); + %positioned_left_etch(); + %positioned_right_etch(); + %positioned_bottom_etch(); + if (include_connector) { + %positioned_top_connector(); + %positioned_bottom_connector(); + } + if (render_bolts) { + positioned_left_bolts(); + positioned_right_bolts(); + positioned_front_bolts(); + } + } + } + + positioned_enclosure(); + if (render_pcb) { + translate([enclosure_wall_to_wall_width + eps, -pcb_hole_to_sensor_x, -magnet_hole_offset - pcb_hole_to_sensor_y]) { + rotate([0, 90, 0]) { + rotate([0, 0, 90]) { + pcb(pcb_to_spool, render_sensor_jig); + translate([0, 0, -thickness - 2 * eps]) { + standard_m4_bolt(nut_distance=thickness + pcb_thickness + 4*eps); + } + } + } + } + } + + translate([spool_width_slop/2 + thickness, 0, 0]) { + // Flap area + if (render_flaps > 0) { + rotate([0, 90, 0]) { + if (render_flap_area >= 1) { + translate([0, 0, thickness]) { + cylinder(r=exclusion_radius, h=flap_width - 2*thickness); + } + } + if (render_flap_area >= 2) { + translate([0, 0, thickness + (flap_width - 2*thickness)/4]) { + cylinder(r=outer_exclusion_radius, h=(flap_width - 2*thickness)/2); + } + } + } + + flap_offset = thickness > flap_notch_depth ? -flap_notch_depth + thickness + flap_width_slop/2 : flap_width_slop/2; + translate([flap_offset, 0, 0]) { + // Collapsed flaps on the top + for (i=[0:num_flaps/2 - 1]) { + if (i == 0 || render_flaps == 2) { + rotate([360/num_flaps * i, 0, 0]) { + translate([flap_width, flap_pitch_radius, 0]) { + rotate([flap_rendered_angle, 0, 180]) { + flap(); + if (i == 0) { + flap_letter(letter, 1); // 1 = top + } + } + } + } + } + } + + // Hanging flaps on the bottom + for (i=[1:num_flaps/2]) { + angle = -360/num_flaps*i; + translate([0, flap_pitch_radius*cos(angle), flap_pitch_radius * sin(angle)]) { + if (i == 1 || render_flaps == 2) { + rotate([-90, 0, 0]) { + flap(); + if (i == 1) { + flap_letter(letter, 2); // 2 = bottom + } + } + } + } + } + } + } + } + + if(render_spool) { + translate([(spool_width_clearance - spool_width + spool_width_slop) / 2 + thickness, 0, 0]) { + spool_struts(); + + // motor spool + translate([spool_width - thickness + 5*spool_horizontal_explosion, 0, 0]) { + rotate([0, 90, 0]) { + color(assembly_color) + flap_spool_complete(motor_shaft=true, magnet_hole=true); + translate([0, 0, -eps - thickness]) + flap_spool_etch(); + } + } + color(assembly_color1) { + translate([thickness - 3*spool_horizontal_explosion, 0, 0]) { + rotate([0, 90, 0]) { + spool_retaining_wall(m4_bolt_hole=true); + } + } + } + translate([-5*spool_horizontal_explosion, 0, 0]) { + rotate([0, 90, 0]) { + color(assembly_color) + flap_spool_complete(captive_nut=true); + flap_spool_etch(); + } + } + translate([thickness * 2, 0, 0]) { + rotate([0, -90, 0]) { + standard_m4_bolt(nut_distance=thickness + 7*spool_horizontal_explosion); + } + } + } + } + + if (render_motor) { + translate([enclosure_wall_to_wall_width - thickness - 28byj48_mount_bracket_height, 0, 0]) { + + rotate([-90, 0, 0]) { + + rotate([0, -90, 0]) { + Stepper28BYJ48(); + } + translate([0, -28byj48_shaft_offset, 0]) { + translate([0, 0, -28byj48_mount_center_offset]) { + rotate([0, 90, 0]) { + rotate([0, 0, 90]) { + standard_m4_bolt(nut_distance=thickness+28byj48_mount_bracket_height); + } + } + } + translate([0, 0, 28byj48_mount_center_offset]) { + rotate([0, 90, 0]) { + rotate([0, 0, 90]) { + standard_m4_bolt(nut_distance=thickness+28byj48_mount_bracket_height); + } + } + } + } + } + } + } +} diff --git a/3d/splitflap_3dprinter.scad b/3d/splitflap_3dprinter.scad new file mode 100644 index 00000000..afefbb5d --- /dev/null +++ b/3d/splitflap_3dprinter.scad @@ -0,0 +1,19 @@ +include; +include; + + +flap_spool_complete(motor_shaft=true, magnet_hole=true); + +translate([0,0,spool_strut_length]) { + rotate([0,90,0]) + spool_struts(); +} + + +translate([60, 0, 0]) { + flap_spool_complete(captive_nut=true); + + translate([0,0, thickness]) { + spool_retaining_wall(m4_bolt_hole=true); + } +} \ No newline at end of file diff --git a/3d/tools/flap_container.scad b/3d/tools/flap_container.scad index ba3f895d..a75c9158 100644 --- a/3d/tools/flap_container.scad +++ b/3d/tools/flap_container.scad @@ -14,7 +14,7 @@ limitations under the License. */ -include <../flap_dimensions.scad> +include <../parts/flap.scad> use <../splitflap.scad> num_flaps = 40; diff --git a/3d/tools/punch_jig.scad b/3d/tools/punch_jig.scad index ec26f9a4..f330c778 100644 --- a/3d/tools/punch_jig.scad +++ b/3d/tools/punch_jig.scad @@ -14,7 +14,7 @@ limitations under the License. */ -include<../flap_dimensions.scad>; +include<../parts/flap.scad>; print_tolerance = 0.1; jig_thickness = 2; diff --git a/3d/tools/scoring_jig.scad b/3d/tools/scoring_jig.scad index ff752a8c..5919fe2b 100644 --- a/3d/tools/scoring_jig.scad +++ b/3d/tools/scoring_jig.scad @@ -14,7 +14,7 @@ limitations under the License. */ -include<../flap_dimensions.scad> +include<../parts/flap.scad> print_tolerance = 0.1; eps = 0.1;