Skip to content
Snippets Groups Projects
HeatSink.scad 8.13 KiB

use <WebWeaver/scad/Util.scad>;
use <WebWeaver/scad/Piping.scad>;



// Material definitions
module GreenPLA() { color([28/255, 1.0, 47/255, 132/255]) children(); }
module MockupColor() { color([0, 0.6, 0.8, 0.6]) children(); }
module BlackAluminium() { color([16/255, 16/255, 16/255]) children(); }





// hs = heat sink
hs_l = 120;
hs_w = 79;
hs_baseplate = 4;

hs_fin_w0 = 2;
hs_fin_w1 = 1.3;
hs_fin_d = 11;  // depth
hs_fin_cnt = 10;
hs_fin_spacing = 3.5;


/**
  *
  * TODO: Moar documentation!
  *
  */
module symmetricHeatsink(
  l = hs_l,
  w = hs_w,
  baseplate = hs_baseplate,
  fin_w0 = hs_fin_w0,
  fin_w1 = hs_fin_w1,
  fin_d = hs_fin_d,
  fin_cnt = hs_fin_cnt,
  fin_spacing = hs_fin_spacing
) {
  
  module fin() {
    dw2 = fin_d - fin_w1/2;

    translate([0, fin_w0, dw2]) rotate([-90, 0, -90]) linear_extrude(l) hull() {
      translate([0, dw2, 0]) square([fin_w0, 0.01]);
      translate([fin_w1/2, 0, 0]) circle(d=fin_w1);
    }
  }
  
  module half() {
    cube([l, w/2, baseplate]);

    for (y=[0:fin_cnt/2 - 1]) translate([0, y * (  fin_w0 + fin_spacing), baseplate]) fin();
  }
  
  BlackAluminium() translate([-l/2, -w/2, 0]) {
    half();
    translate([0, w, 0]) mirror([0, 1, 0]) half();
  }
}


// f = fan
f_r = 4;  // corner radius of fan case
f_d = 20;  // depth. or thickness.
f_wh = 40;  // widht and height of the fan's chassis
f_drill = 4;  // mounting hole diameter
f_drill_wh = 32;  // size of a virtual rectangle with mounting holes drilled through the corners
f_intake = 38;  // air intake diameter
f_motor = 23;  // diameter of fan motor


/**
  *
  * TODO: Moar documentation!!!
  * 
  * `layer`: either "Body" or "MotorConnection". the latter produces the bridges to the motor
  *
  */
module fanProfile(
  r = f_r,
  wh = f_wh,
  drill = f_drill,
  drill_wh = f_drill_wh,
  intake = f_intake,
  motor = f_motor,

  layer = "Body",
  closed=false
) {
  
  difference() {
    roundRect(wh, wh, r, center=true);
    
    // mounting holes
    _dwh2 = drill_wh / 2;
    for (tx=[-_dwh2, _dwh2], ty=[-_dwh2, _dwh2]) translate([tx, ty, 0]) circle(d=drill);
    
    // the central air intake
    if (!closed) circle(d=intake);
  }

  // the fans motor
  if (!closed && motor > 0) circle(d=motor);
}


/**
  *
  * TODO: Moar documentation!!!
  *
  * `d` = thickness
  *
  */
module fan(
  d = f_d,
  r = f_r,
  wh = f_wh,
  drill = f_drill,
  drill_wh = f_drill_wh,
  intake = f_intake,
  motor = f_motor,
  layer_ratio = 0.1
) {

  // shadow original module here so we don't have to repeat the param list ^^
  module _fanProfile(layer) {
    fanProfile(r=r, wh=wh, drill=drill, drill_wh=drill_wh, intake=intake, motor=motor, layer=layer);
  }

  if (layer_ratio > 0) translate([0, 0, d - d * layer_ratio]) linear_extrude(d * layer_ratio) _fanProfile("MotorConnection");

  if (layer_ratio < 1) linear_extrude(d * (1 - layer_ratio)) _fanProfile("Body");  
}


fu_strength = 2;  // material strength

fu_outlet_w = (hs_fin_cnt / 2) * hs_fin_w0 + (hs_fin_cnt / 2 - 1) * hs_fin_spacing;  // Σ of fins spacing and width calculated over one side;


module tri(w, h) {
  polygon([
    [0, -h/2],
    [0, h/2],
    [w, 0]
  ]);
}


// length of the outlet
x = 5;  // TODO: rename!!!!

// The profile of the funnel, ready for extrusion
module funnel2D() {
  
  _y = hs_w/2 - fu_outlet_w;

  difference() {
    
    union() {
      translate([0, -hs_w/2, 0]) square([x, hs_w]);
      translate([0, 0, 0]) roundTri(x*9, hs_w, 5);
    }
    
    /* union() {
      translate([0, _y, 0]) square([x, fu_outlet_w]);
      translate([0, -_y - fu_outlet_w, 0]) square([x, fu_outlet_w]);
      
      // air funnel split
      translate([x, 0, 0]) tri(_y*4, hs_w);
    } */

    // air funnel split
    translate([0, 0, 0]) roundTri(x*4, _y*2, 3);
    translate([0, -_y, 0]) square([x/2, _y*2]);
  }
}



// ff = funnel fin
ff_str = hs_fin_w0;  // strength

module funnelFins(
  w = hs_w,
  fin_cnt = hs_fin_cnt,
  fin_spacing = hs_fin_spacing,
  fin_cnt = hs_fin_cnt,
  fin_w0 = hs_fin_w0,
  _str = ff_str
) {
  
  module fin(x, y) {
    module _fF(y) {
      translate([0, -y/2, 0]) square([x, y]);
      translate([0, 0, 0]) roundTri((x*9) * (y/w), y, 5);
    }

    difference() {
      _fF(y + _str/2);
      _fF(y - _str/2);
    }
  }
  
  _x = 5;
  for (y=[0:fin_cnt/2 - 1]) fin( -(y/fin_cnt/2 - 1) * _x,  w - y * (fin_w0 + fin_spacing) * 2);
}




// ############################
// ####### New Approach #######
// ############################



fu_r = 2;
fu_str = 1.6;

module curvyFunnel() {
    
  hs_h = hs_baseplate + hs_fin_d;

  d = 2;
    
  y_off = hs_w/2 - fu_outlet_w/2;
    
  module half() {
    difference() {
      union() {
        translate([hs_l/2, -y_off, -hs_h]) rotate([90, 0, 90]) cornerPipeLayOnFace(hs_h, fu_outlet_w, d, 180, fu_r, fu_str, centerX=true);
    
        translate([hs_l/2, -y_off + fu_outlet_w/2 + d/2, d]) rotate([0, 0, 180]) cornerPipe(fu_outlet_w, hs_h, d, 90, fu_r, fu_str);
    
        translate([hs_l/2 - fu_outlet_w - d, -y_off + fu_outlet_w/2 + d/2, d]) rotate([0, 0, 0]) cornerPipe(fu_outlet_w, hs_h, d, 90, fu_r, fu_str);
      }

      translate([-hs_l/2, 0, 0]) cube([hs_l, hs_w/2, hs_w]);
    }
  }


  GreenPLA() {
    half();
    mirror([0, 1, 0]) half();
  }
}


module splitFunnel() {
    
  hs_h = hs_baseplate + hs_fin_d;

  d = 2;
    
  y_off = hs_w/2 - fu_outlet_w/2;
  
  alpha = 56;  // curve angle
    
  module half() {

    translate([hs_l/2, -y_off, -hs_h]) rotate([90, 0, 90]) cornerPipeLayOnFace(hs_h, fu_outlet_w, d, 180, fu_r, fu_str, centerX=true);

    // intersection() {  // debugging stuff

    translate([hs_l/2, -y_off + fu_outlet_w/2 + d/2, d]) rotate([0, 0, 270 - alpha]) cornerPipe(fu_outlet_w, hs_h, d, alpha, fu_r, fu_str);


    // the straight pipe after the curve
    
    sp_len = 20;  // TODO: calculate!
    
    // translate([hs_l/2, y_off, hs_h/2 + d]) rotate([90, 0, -90]) straightPipe(fu_outlet_w, hs_h, sp_len, fu_r, fu_str);
    
    
    
    
    


//    difference() {
//      translate([-hs_l/2, 0, 0]) cube([hs_l, hs_w/2, hs_w]);
//    }
  }
  
  sp_len = 20;

  module _straight(x_off=0, _y_off=0, inner=false, outer=false) {
    // connecting a straight pipe to the end of a curve in a fully generic way
    //
    // step 1: move to a very specific point beneath origin
    // step 2: rotate by curve pipe angle
    // step 3: move to the beginning(!) of the curve pipe  where we want them to meet
    //
    for (signum=[-1, 1])
    translate([hs_l/2, (-y_off + fu_outlet_w/2) * signum, hs_h/2 + d])         // step 3
      rotate([0, 0, -alpha * signum *$t])                                        // step 2
      translate([+ x_off, -fu_outlet_w/2 * signum + d/2 + _y_off, 0]) rotate([90, 0, -90])        // step 1
      straightPipe(fu_outlet_w, hs_h, sp_len + 2*x_off, fu_r, fu_str, inner=inner, outer=outer);

    // }  // debugging stuff
  }
  
  color([1.0, 0, 0]) translate([0, 0, 0.1]) _straight(_y_off=-1, outer=true);

  GreenPLA() {    
    half();
    mirror([0, 1, 0]) half();


    difference() {
      union() {
        translate([hs_l/2 - f_wh/2, 0, 0]) rotate([0, 0, 45]) linear_extrude(hs_h + d) roundRect(f_wh, f_wh, f_r, center=true);
        //_straight(outer=true);
      }
      
      translate([hs_l/2 - f_wh/2, 0, d + fu_str]) rotate([0, 0, 45]) cylinder(d=30, h=hs_h - fu_str);


      _straight(x_off=1, inner=true);
    }


  }
}



// Define the main module…
module HeatSinkMainModule() {
  
  epsilon = 0.001;
  
  if ($preview) {
    
    $fs = 0.5;
    $fa = 10;

    // display heat sink only in render mode
    rotate([180, 0, 0]) symmetricHeatsink();
    
    
    // translate([0, 0, 0]) curvyFunnel();  
    translate([0, 0, 0]) splitFunnel();

    // color([0, 0, 1.0, 1.0]) translate([0, 0, 1]) linear_extrude(1) funnelFins();

    translate([7+8 - 72/2, 0, epsilon]) rotate([0, 0, 30]) MockupColor() linear_extrude(1.6) roundNGon(72, 6, 10/2);
    
    // translate([12, 0, f_wh/2 + epsilon]) rotate([0, 90, 0]) MockupColor() fan(d=11);

  } else {

    $fs = 0.35;
    $fa = 2.0;
    
    // TODO: 3D-Print positioning!

    // translate([0, 0, 0]) curvyFunnel();  
    translate([0, 0, 0]) splitFunnel();

    // color([0, 0, 1.0, 1.0]) translate([0, 0, 1]) linear_extrude(1) funnelFins();
  }
}

// …and execute it ;)
HeatSinkMainModule();