/*
* Trackball
* Copyright 2022 Thomas Buck - thomas@xythobuz.de
*
* Required parts:
* - 1x Raspberry Pi Pico
* - 4x Cherry MX compatible switches and keycaps
* - 1x Billard ball, diameter 38mm
* - 3x Si3N4 static bearing balls, diameter 3mm
* - 1x PMW3360 sensor with breakout board
* - 8x M2 screw, length 5mm
* - 8x M2 heat melt insert, length 4mm
*
* For the PMW3360 breakout board get this:
* https://github.com/jfedor2/pmw3360-breakout
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* See .
*/
// https://www.thingiverse.com/thing:421524
use
// https://www.printables.com/model/210898-raspberry-pi-pico-case
use
// ######################
// ## Rendering Select ##
// ######################
//ball_and_roller();
//pico_wrap();
//sensor();
//mx_switch_cutout(wall);
//mx_switch_test();
//roller_mount_test();
//roller_holder();
//roller_mount_tri();
//trackball_top();
//trackball_bottom();
assembly();
//print();
// #######################
// #### Configuration ####
// #######################
$fn = 200;//42;
ball_dia = 38.0;
roller_dia = 3.0;
roller_ball_h = ball_dia / 2 - 5;
roller_count = 3;
wall = 3.0;
$c = 0.1;
$e = 0.01;
left_hand_version = false;
cut_roller_holder = false;
draw_supports = false;
draw_ball_roller = true;
draw_switches = true;
draw_sensor = true;
use_external_pico_model = true;
// #######################
// ## Raspberry Pi Pico ##
// #######################
pico_w = 21;
pico_l = 51;
pico_d = 1.0;
pico_hole_d = 2.1;
pico_hole_x = 4.8;
pico_hole_y = 2.0;
pico_hole_d_x = 11.4;
pico_hole_d_y = pico_l - 2 * pico_hole_y;
pico_usb_w = 8.0;
pico_usb_h = 2.8;
pico_usb_d = 6.0;
pico_usb_off = 1.3;
// ######################
// ### PMW3360 Sensor ###
// ######################
// https://github.com/jfedor2/pmw3360-breakout
sensor_w = 22;
sensor_l = 34;
sensor_pcb_h = 1.6;
sensor_hole_dia = 2.2;
sensor_hole_off_x = 3.0;
sensor_hole_off_y = 3.0;
sensor_hole_dist_x = 16.0;
sensor_hole_dist_y = 24.5;
sensor_cut_w = 8.0 + 0.5;
sensor_cut_h = 17.26;
sensor_cut_off_x = 7.0 - 0.25;
sensor_cut_off_y = 5.27;
sensor_cut_edge_to_pin1 = 2.75;
sensor_edge_to_pin1 = 1.52;
sensor_ball_to_lens_top = 2.4;
sensor_ball_to_chip_bottom = 9.81;
sensor_chip_w = 9.1;
sensor_chip_l = 16.2;
sensor_chip_h = 2.21;
sensor_pin_w = 0.5;
sensor_pin_h = 4.51;
sensor_pin_d = 0.2;
sensor_pin_dist = 10.7;
sensor_pin_off_top = 0.5;
sensor_pin_pitch = 0.89;
sensor_pin1_to_optical_center = 5.66;
sensor_lens_cutout_r = 2.0;
sensor_lens_cutout_w = 4.0;
sensor_lens_cutout_growth = 0.25;
sensor_lens_cutout_to_chip = 6.71 - 1.60;
sensor_lens_baseplate_h = 2.40;
sensor_lens_d = 19.0 + 1.0;
sensor_lens_w = 21.35 + 0.2;
sensor_lens_off = 10.97;
// ######################
// ## MX Switch Cutout ##
// ######################
// https://geekhack.org/index.php?topic=70654.0
mx_co_w = 14.0;
mx_co_w_add = 0.8;
mx_co_h = 14.0;
mx_co_h_off_1 = 1.0;
mx_co_h_off_2 = 3.5;
mx_co_h_off_3 = mx_co_h - 2 * (mx_co_h_off_1 + mx_co_h_off_2);
mx_co_r = 0.4;
// https://geekhack.org/index.php?topic=71550.0
mx_co_th = 1.5 - 0.1;
mx_co_b_add = 1.0;
mx_co_b_w = mx_co_w + mx_co_b_add;
mx_co_b_h = mx_co_h + mx_co_b_add;
mx_travel = 3.9;
// ######################
// ### Implementation ###
// ######################
base_dia = 62;
grub_screw_dia = 2.8;
grub_channel_dia = 4.0;
roller_thread_dia = roller_dia + 5.0;
roller_h = roller_dia + 7.0;
roller_ball_h_off = 0.4;
roller_ball_hold_off = 0.5;
roller_thread_hole = roller_dia - 1;
roller_small_hole = sphere_r_at_h(roller_ball_hold_off, roller_dia / 2) * 2;
roller_ridge_h = 1.5;
roller_mount_angle_off = 90;
roller_mount_dia = roller_thread_dia + 2.0;
ball_h = 15; // todo
switch_test_w = 25;
roller_mount_holder_gap = 0.8;
sensor_pcb_mount_gap = 2.0;
sensor_pcb_support_h = 1.6 + 3.4;
sw1_rot_x = -5;
sw1_rot_z = -30 * (left_hand_version ? -1 : 1);
sw1_trans_x = 0;
sw1_trans_y = -base_dia / 2 - 0;
sw1_trans_z = -17;
sw2_rot_x = -5;
sw2_rot_z = 15 * (left_hand_version ? -1 : 1);
sw2_trans_x = 0;
sw2_trans_y = -base_dia / 2 - 0;
sw2_trans_z = -17;
sw3_rot_x = -5;
sw3_rot_z = 125 * (left_hand_version ? -1 : 1);
sw3_trans_x = 0;
sw3_trans_y = -base_dia / 2 - 0;
sw3_trans_z = -17;
sw4_rot_x = -5;
sw4_rot_z = -125 * (left_hand_version ? -1 : 1);
sw4_trans_x = 0;
sw4_trans_y = -base_dia / 2 - 0;
sw4_trans_z = -17;
sw_mount_w = mx_co_w + 7;
sw_mount_co_l = 10;
bottom_base_wall = wall + 0.5;
bottom_base_below_zero = bottom_base_wall + 4.5;
pico_co_w = pico_w + 1;
pico_co_l = pico_l + 1;
reset_button_dia = 4.0;
reset_button_off_x = 7;
reset_button_off_y = 12.15;
pico_support_w = 6.5;
pico_support_l = 5;
pico_screw_depth = 6;
pico_screw_d = 1.8;
usb_cutout_grow_l = 10;
usb_cutout_grow_x = 20;
usb_cutout_grow_y = 40;
usb_cutout_w_add = 1;
usb_cutout_h_add = 0.6;
assembly_dist = 20;
roller_holder_sider_cut = 1.5;
roller_holder_h_compensation = -0.1;
bottom_add_wall = 4;
screw_dia = 3.2;
screw_off = base_dia / 2 - 10;
screw_head_d = 6.0;
screw_head_h = 3.5;
screw_angles = [ 15, -15, 180 + 15, 180 - 15 ];
screw_insert_dia = 4.8;
screw_insert_h = 6.0;
function sphere_r_at_h(h, r) = r * sin(acos(h / r));
function sphere_angle_at_rh(h, r) = acos(h / r);
module mx_switch_cutout(h) {
translate([-mx_co_w / 2 - mx_co_w_add, -mx_co_h / 2, 0]) {
linear_extrude(h + 1) {
translate([mx_co_w_add, 0]) {
square([mx_co_w, mx_co_h]);
for (x = [mx_co_r / 2, mx_co_w - mx_co_r / 2])
for (y = [mx_co_r / 2, mx_co_h - mx_co_r / 2])
translate([x, y])
circle(r = mx_co_r);
}
for (x = [0, mx_co_w + mx_co_w_add])
for (y = [0, mx_co_h_off_2 + mx_co_h_off_3])
translate([x, mx_co_h_off_1 + y, 0])
square([mx_co_w_add, mx_co_h_off_2]);
}
translate([mx_co_w_add - mx_co_b_add / 2, -mx_co_b_add / 2, -1])
cube([mx_co_b_w, mx_co_b_h, h - mx_co_th + 1]);
}
}
module mx_switch_test() {
difference() {
translate([-switch_test_w / 2, -switch_test_w / 2, 0])
cube([switch_test_w, switch_test_w, wall]);
mx_switch_cutout(wall);
translate([0, -switch_test_w / 2 + 1, wall - 1.0])
linear_extrude(1.1)
text("switch test", size = 3, halign = "center");
}
%translate([0, 0, wall])
rotate([0, 0, 180])
mx_switch($t);
}
module pico_own() {
translate([-pico_w / 2, -pico_l / 2, 0])
difference() {
union() {
color("green")
cube([pico_w, pico_l, pico_d]);
translate([(pico_w - pico_usb_w) / 2, pico_l - pico_usb_d + pico_usb_off, pico_d])
cube([pico_usb_w, pico_usb_d, pico_usb_h]);
}
for (x = [0, pico_hole_d_x])
for (y = [0, pico_hole_d_y])
translate([pico_hole_x + x, pico_hole_y + y, -1])
cylinder(d = pico_hole_d, h = pico_d + 2);
}
}
module pico_wrap() {
//if (use_external_pico_model)
translate([-pico_w / 2, -pico_l / 2, 0])
pico();
//else
pico_own();
}
module sensor_lens_cutout_intern() {
cylinder(d = sensor_lens_cutout_r * 2, h = $e);
translate([-sensor_lens_cutout_r, 0, 0])
cube([sensor_lens_cutout_r * 2, sensor_lens_cutout_w, $e]);
}
module rounded_cube(x, y, z, r) {
hull()
for (tx = [r, x - r])
for (ty = [r, y - r])
translate([tx, ty, 0])
cylinder(d = r * 2, h = z);
}
module sensor_lens_cutout() {
translate([0, 0, sensor_lens_cutout_to_chip])
hull() {
translate([0, 0, sensor_lens_baseplate_h - $e])
sensor_lens_cutout_intern();
scale(1 + sensor_lens_cutout_growth * sensor_lens_baseplate_h)
sensor_lens_cutout_intern();
}
translate([-sensor_lens_d / 2, -sensor_lens_w + sensor_lens_off, 0])
rounded_cube(sensor_lens_d, sensor_lens_w, sensor_lens_cutout_to_chip, 6);
translate([-3 / 2, -sensor_lens_w + sensor_lens_off - 0.5, 0])
cube([3, 0.5, sensor_lens_cutout_to_chip]);
}
module sensor() {
translate([-sensor_w / 2, -sensor_l / 2, 0])
difference() {
color("green")
cube([sensor_w, sensor_l, sensor_pcb_h]);
translate([sensor_cut_off_x, sensor_cut_off_y, -1])
cube([sensor_cut_w, sensor_cut_h, sensor_pcb_h + 2]);
for (x = [0, sensor_hole_dist_x])
for (y = [0, sensor_hole_dist_y])
translate([sensor_hole_off_x + x, sensor_hole_off_y + y, -1])
cylinder(d = sensor_hole_dia, h = sensor_pcb_h + 2);
}
color("#303030")
translate([-sensor_chip_w / 2, -sensor_l / 2 - sensor_chip_l + sensor_edge_to_pin1 + sensor_cut_off_y + sensor_cut_h - sensor_cut_edge_to_pin1, -sensor_chip_h])
cube([sensor_chip_w, sensor_chip_l, sensor_chip_h]);
translate([0, -sensor_l / 2 - 15 * sensor_pin_pitch + sensor_cut_off_y + sensor_cut_h - sensor_cut_edge_to_pin1, 0])
for (p = [0 : 15])
translate([0, p * sensor_pin_pitch, 0])
for (x = [-sensor_pin_dist / 2, sensor_pin_dist / 2])
if (((p % 2 == 0) && (x < 0))
|| ((p % 2 == 1) && (x > 0)))
translate([-sensor_pin_d / 2 + x, -sensor_pin_w / 2, -sensor_chip_h + sensor_pin_off_top])
cube([sensor_pin_d, sensor_pin_w, sensor_pin_h]);
translate([0, -sensor_l / 2 + sensor_cut_off_y + sensor_cut_h - sensor_cut_edge_to_pin1 - sensor_pin1_to_optical_center, 0]) {
color("cyan")
translate([0, 0, -sensor_chip_h + 1])
cylinder(d = 0.2, h = sensor_ball_to_chip_bottom - 1);
%color("blue")
sensor_lens_cutout();
}
}
module ball_and_roller() {
color("red")
sphere(d = ball_dia, $fn = $fn * 2);
for (r = [0 : roller_count - 1])
rotate([0, 0, roller_mount_angle_off + 360 / roller_count * r])
translate([sphere_r_at_h(roller_ball_h - ball_dia / 2, ball_dia / 2), 0, -ball_dia / 2 + roller_ball_h])
rotate([0, 180 + sphere_angle_at_rh(roller_ball_h - ball_dia / 2, ball_dia / 2), 0])
translate([0, 0, -roller_dia / 2])
roller_holder();
}
module roller_holder() {
translate([0, 0, -roller_h + roller_dia / 2])
difference() {
color("magenta")
union() {
translate([0, 0, roller_h-roller_dia/2 + roller_ball_h_off-3])
cylinder(d1 = roller_mount_dia, d2=roller_dia+1, h = 3);
cylinder(d = roller_mount_dia, h = roller_h-roller_dia/2 + roller_ball_h_off-3);
}
translate([-roller_mount_dia / 2 - 1, roller_mount_dia / 2 - roller_holder_sider_cut, -1])
cube([roller_mount_dia + 2, roller_mount_dia / 2 + 1, roller_h + 2]);
translate([0, 0, -$e])
cylinder(d = roller_thread_hole, h = $e+ roller_h - roller_dia / 2 + roller_ball_h_off + roller_ball_hold_off);
translate([0, 0, roller_h - roller_dia / 2 + roller_holder_h_compensation])
sphere(d = roller_dia, $fn = $fn * 2);
if (cut_roller_holder)
translate([-roller_thread_dia / 2 - 1, -roller_thread_dia, -1])
cube([roller_thread_dia + 2, roller_thread_dia, roller_h + 2]);
}
%color("blue")
sphere(d = roller_dia, $fn = $fn * 2);
}
module roller_mount() {
translate([0, 0, -1-roller_h + roller_dia / 2]) {
difference() {
cylinder(d=roller_mount_dia+wall,h=roller_h/2);
translate([0, 0, 1])
cylinder(d=roller_mount_dia+$c*2,h=roller_h/2+$e);
if (cut_roller_holder)
translate([-roller_thread_dia / 2 - 1, -roller_thread_dia, -1])
cube([roller_thread_dia + 2, roller_thread_dia, roller_h + 2]);
}
}
}
module roller_mount_test() {
roller_holder();
roller_mount();
}
module roller_mount_tri_hull() {
for (r = [0 : roller_count - 1])
rotate([0, 0, roller_mount_angle_off + 360 / roller_count * r])
translate([sphere_r_at_h(roller_ball_h - ball_dia / 2, ball_dia / 2), 0, -ball_dia / 2 + roller_ball_h])
rotate([0, 180 + sphere_angle_at_rh(roller_ball_h - ball_dia / 2, ball_dia / 2), 0])
translate([0, 0, -roller_h])
cylinder(d = roller_mount_dia + wall + 1, h = roller_h - 3);
translate([0, 0, -ball_dia / 2 - 11])
cylinder(d = base_dia, h = $e);
}
module roller_mount_tri_body() {
// space for roller holder
for (r = [0 : roller_count - 1])
rotate([0, 0, roller_mount_angle_off + 360 / roller_count * r])
translate([sphere_r_at_h(roller_ball_h - ball_dia / 2, ball_dia / 2), 0, -ball_dia / 2 + roller_ball_h])
rotate([0, 180 + sphere_angle_at_rh(roller_ball_h - ball_dia / 2, ball_dia / 2), 0])
translate([0, 0, -roller_h])
cylinder(d = roller_mount_dia + roller_mount_holder_gap, h = ball_dia / 2 + roller_h);
// room for ball itself
sphere($fn = $fn * 2, d = ball_dia + $c * 2 + 4);
// grub screws
for (r = [0 : roller_count - 1])
rotate([0, 0, roller_mount_angle_off + 360 / roller_count * r])
translate([sphere_r_at_h(roller_ball_h - ball_dia / 2, ball_dia / 2), 0, -ball_dia / 2 + roller_ball_h])
rotate([0, 180 + sphere_angle_at_rh(roller_ball_h - ball_dia / 2, ball_dia / 2), 0])
translate([0, 0, -roller_h/2])
rotate([0,-90,0])
translate([-2, 0, 2]) {
cylinder(d = grub_screw_dia, h = ball_dia);
translate([0, 0, roller_mount_dia / 4 + wall])
cylinder(d = grub_channel_dia, h = ball_dia);
}
// sensor lens
translate([0, 0, -ball_dia / 2 - ball_h])
translate([0, sensor_l / 2 - sensor_cut_off_y - sensor_cut_h + sensor_cut_edge_to_pin1 + sensor_pin1_to_optical_center, ball_h + sensor_chip_h - sensor_ball_to_chip_bottom])
translate([0, -sensor_l / 2 + sensor_cut_off_y + sensor_cut_h - sensor_cut_edge_to_pin1 - sensor_pin1_to_optical_center, 0])
sensor_lens_cutout();
// sensor pcb
translate([-1, -1, -ball_dia / 2 - ball_h])
translate([0, sensor_l / 2 - sensor_cut_off_y - sensor_cut_h + sensor_cut_edge_to_pin1 + sensor_pin1_to_optical_center, ball_h + sensor_chip_h - sensor_ball_to_chip_bottom])
translate([-sensor_w / 2, -sensor_l / 2, -10])
cube([sensor_w + 2, sensor_l + 2, sensor_pcb_h + 10 + sensor_pcb_mount_gap]);
}
module roller_mount_sensor_pcb_support() {
translate([-sensor_w / 2, -sensor_l / 2, sensor_pcb_h])
translate([0, 0, -ball_dia / 2 - ball_h])
translate([0, sensor_l / 2 - sensor_cut_off_y - sensor_cut_h + sensor_cut_edge_to_pin1 + sensor_pin1_to_optical_center, ball_h + sensor_chip_h - sensor_ball_to_chip_bottom])
for (x = [0, sensor_hole_dist_x])
for (y = [0, sensor_hole_dist_y])
translate([sensor_hole_off_x + x, sensor_hole_off_y + y, 0])
difference() {
union() {
color("magenta")
cylinder(d = sensor_hole_dia + 1.5, h = sensor_pcb_mount_gap);
if (draw_supports)
color("black")
translate([0, 0, -sensor_pcb_support_h])
cylinder(d = sensor_hole_dia + 0.5, h = sensor_pcb_support_h);
}
cylinder(d = sensor_hole_dia - 0.2, h = sensor_pcb_mount_gap + 1);
}
if (draw_supports)
color("black")
for (x = [-5, 0, 5])
for (y = [-8, 0, 6.5])
if (((x == 0) && (y != 0)) || ((x != 0) && (y == 0)))
translate([x, y + 2, -30])
cylinder(d = sensor_hole_dia + 0.5, h = 8.5);
if (draw_sensor)
%translate([0, 0, -ball_dia / 2 - ball_h])
translate([0, sensor_l / 2 - sensor_cut_off_y - sensor_cut_h + sensor_cut_edge_to_pin1 + sensor_pin1_to_optical_center, ball_h + sensor_chip_h - sensor_ball_to_chip_bottom])
sensor();
}
// TODO holes for pcb screws not going into body!!
module roller_mount_tri() {
if (draw_ball_roller)
%ball_and_roller();
difference() {
hull()
roller_mount_tri_hull();
roller_mount_tri_body();
// TODO test cable cutout
translate([-6, 0, -30.1])
cube([12, 50, 2]);
if (cut_roller_holder)
translate([0, -base_dia / 2 - 1, -40])
cube([base_dia / 2 + 1, base_dia + 2, 40]);
}
roller_mount_sensor_pcb_support();
}
module trackball_top() {
translate([0, 0, ball_dia / 2 + ball_h]) {
if (draw_ball_roller)
%ball_and_roller();
difference() {
color("orange")
hull() {
roller_mount_tri_hull();
rotate([0, 0, sw1_rot_z])
translate([sw1_trans_x, sw1_trans_y, sw1_trans_z])
rotate([90 + sw1_rot_x, 0, 0])
translate([0, 0, -0.5])
cube([sw_mount_w, sw_mount_w, 1], center = true);
rotate([0, 0, sw2_rot_z])
translate([sw2_trans_x, sw2_trans_y, sw2_trans_z])
rotate([90 + sw2_rot_x, 0, 0])
translate([0, 0, -0.5])
cube([sw_mount_w, sw_mount_w, 1], center = true);
rotate([0, 0, sw3_rot_z])
translate([sw3_trans_x, sw3_trans_y, sw3_trans_z])
rotate([90 + sw3_rot_x, 0, 0])
translate([0, 0, -0.5])
cube([sw_mount_w, sw_mount_w, 1], center = true);
rotate([0, 0, sw4_rot_z])
translate([sw4_trans_x, sw4_trans_y, sw4_trans_z])
rotate([90 + sw4_rot_x, 0, 0])
translate([0, 0, -0.5])
cube([sw_mount_w, sw_mount_w, 1], center = true);
}
roller_mount_tri_body();
if (cut_roller_holder)
translate([0, -base_dia / 2 - 1, -40])
cube([base_dia / 2 + 1, base_dia + 2, 40]);
rotate([0, 0, sw1_rot_z])
translate([sw1_trans_x, sw1_trans_y, sw1_trans_z])
rotate([90 + sw1_rot_x, 0, 0])
translate([0, 0, -sw_mount_co_l]) {
mx_switch_cutout(sw_mount_co_l + 1);
translate([0, 0, 2])
rotate([90, 0, 0])
cylinder(d = 4, h = 20);
}
rotate([0, 0, sw2_rot_z])
translate([sw2_trans_x, sw2_trans_y, sw2_trans_z])
rotate([90 + sw2_rot_x, 0, 0])
translate([0, 0, -sw_mount_co_l]) {
mx_switch_cutout(sw_mount_co_l + 1);
translate([0, 0, 2])
rotate([90, 0, 0])
cylinder(d = 4, h = 20);
}
rotate([0, 0, sw3_rot_z])
translate([sw3_trans_x, sw3_trans_y, sw3_trans_z])
rotate([90 + sw3_rot_x, 0, 0])
translate([0, 0, -sw_mount_co_l]) {
mx_switch_cutout(sw_mount_co_l + 1);
translate([0, 0, 2])
rotate([90, 0, 0])
cylinder(d = 4, h = 20);
}
rotate([0, 0, sw4_rot_z])
translate([sw4_trans_x, sw4_trans_y, sw4_trans_z])
rotate([90 + sw4_rot_x, 0, 0])
translate([0, 0, -sw_mount_co_l]) {
mx_switch_cutout(sw_mount_co_l + 1);
translate([0, 0, 2])
rotate([90, 0, 0])
cylinder(d = 4, h = 20);
}
/*
hull() {
rotate([0, 0, sw1_rot_z])
translate([sw1_trans_x, sw1_trans_y, sw1_trans_z])
rotate([90 + sw1_rot_x, 0, 0])
translate([0, 0, -sw_mount_co_l + 5])
mx_switch_cutout(sw_mount_co_l - 10);
rotate([0, 0, sw2_rot_z])
translate([sw2_trans_x, sw2_trans_y, sw2_trans_z])
rotate([90 + sw2_rot_x, 0, 0])
translate([0, 0, -sw_mount_co_l + 5])
mx_switch_cutout(sw_mount_co_l - 10);
}
*/
for (r = screw_angles)
rotate([0, 0, r])
translate([screw_off, 0, -ball_dia / 2 - 11 -1]) {
cylinder(d = screw_insert_dia, h = screw_insert_h + 1);
}
}
roller_mount_sensor_pcb_support();
if (draw_switches) {
%rotate([0, 0, sw1_rot_z])
translate([sw1_trans_x, sw1_trans_y, sw1_trans_z])
rotate([90 + sw1_rot_x, 0, 0])
mx_switch($t);
%rotate([0, 0, sw2_rot_z])
translate([sw2_trans_x, sw2_trans_y, sw2_trans_z])
rotate([90 + sw2_rot_x, 0, 0])
mx_switch($t);
%rotate([0, 0, sw3_rot_z])
translate([sw3_trans_x, sw3_trans_y, sw3_trans_z])
rotate([90 + sw3_rot_x, 0, 0])
mx_switch($t);
%rotate([0, 0, sw4_rot_z])
translate([sw4_trans_x, sw4_trans_y, sw4_trans_z])
rotate([90 + sw4_rot_x, 0, 0])
mx_switch($t);
}
}
}
module trackball_bottom_wrap() {
%rotate([0, 180, 0])
pico_wrap();
color("magenta")
translate([0, 0, -bottom_base_below_zero])
difference() {
cylinder(d = base_dia, h = bottom_base_below_zero + ball_h - 11);
translate([0, 0, bottom_base_wall])
cylinder(d = base_dia - bottom_base_wall * 2 - bottom_add_wall, h = bottom_base_below_zero + ball_h - 11);
translate([-pico_co_w / 2, -pico_co_l / 2, bottom_base_wall])
cube([pico_co_w, pico_co_l, bottom_base_below_zero + ball_h - 11]);
translate([pico_w / 2 - reset_button_off_x, pico_l / 2 - reset_button_off_y, -1])
cylinder(d = reset_button_dia, h = bottom_base_wall + 2);
if (cut_roller_holder)
translate([-base_dia / 2 - 1, -base_dia / 2 - 1, -10])
cube([base_dia / 2 + 1, base_dia + 2, 40]);
}
color("cyan")
for (x = [-1, 1])
for (y = [-1, 1])
translate([x * (pico_co_w - pico_support_w) / 2, y * (pico_co_l - pico_support_l) / 2, 0])
translate([-pico_support_w / 2, -pico_support_l / 2, -bottom_base_below_zero + bottom_base_wall])
cube([pico_support_w, pico_support_l, bottom_base_below_zero - bottom_base_wall - pico_d]);
color("cyan")
for (r = screw_angles)
rotate([0, 0, r])
translate([screw_off, 0, -bottom_base_below_zero + bottom_base_wall])
cylinder(d = screw_head_d + 4, h = bottom_base_below_zero + ball_h - 11 - bottom_base_wall);
}
module usb_cutout() {
hull() {
translate([-usb_cutout_w_add / 2, 0, -usb_cutout_h_add / 2])
cube([pico_usb_w + usb_cutout_w_add, 1, pico_usb_h + usb_cutout_h_add]);
translate([-usb_cutout_grow_x / 2, usb_cutout_grow_l, -usb_cutout_grow_y / 2])
cube([pico_usb_w + usb_cutout_grow_x, 1, pico_usb_h + usb_cutout_grow_y]);
}
}
module trackball_bottom() {
difference() {
trackball_bottom_wrap();
for (x = [0, pico_hole_d_x])
for (y = [0, pico_hole_d_y])
translate([-pico_w / 2, -pico_l / 2, 0])
translate([pico_hole_x + x, pico_hole_y + y, -pico_d - pico_screw_depth])
cylinder(d = pico_screw_d, h = pico_d + pico_screw_depth + 1);
rotate([0, 180, 0])
translate([-pico_w / 2, -pico_l / 2, 0])
translate([(pico_w - pico_usb_w) / 2, pico_l - 1 + pico_usb_off, pico_d])
usb_cutout();
for (r = screw_angles)
rotate([0, 0, r])
translate([screw_off, 0, -bottom_base_below_zero - 1]) {
cylinder(d = screw_dia, h = bottom_base_below_zero + 30);
cylinder(d = screw_head_d, h = screw_head_h + 1);
}
}
}
module assembly() {
translate([0, 0, assembly_dist / 2])
trackball_top();
translate([0, 0, -assembly_dist / 2])
trackball_bottom();
}
module print() {
translate([-40, 0, -4])
trackball_top();
translate([40, 0, bottom_base_below_zero])
trackball_bottom();
for (y = [-20, 0, 20])
translate([0, y, 8.5])
roller_holder();
}