2018年6月16日土曜日

小型工作機CNC2418 その54(Joystickコントローラー始動)

前々回のブレッドボード検討前回のケース加工に引き続き
CandleのJoystick Cotrollerの製作の組立と実働チェックです。
Candleとは、GRBL Controllerで、GitHubのCandle v.1.1.7のことです。
役目としてはGRBL Senderが正しいのかもですが。

まずは、SparkFun Pro Microにつけるピンヘッダー。
L型のピンヘッダーの持ちわせがないので、ストレートの先を折り曲げます。
SparkFun Pro Microにピンヘッダーをハンダ付けします。
ハンダゴテ新調したのです!
小手先5本、予備セラミックヒーター予備付きで、25%OFF US$ 6.22
噂によると、温度コントロールの温度目盛りは、あてにならないらしいので
溶け具合で自分なりの位置を覚えておく必要があるようです。
小手先の劣化が激しいらしいので使う時だけ温度を上げるなど工夫も必要そうですが
軽くて小さいのに60Wと強力なので、立ち上がりも早くなかなか使い易い印象です。
アフィリンクはなく直リンなのでその内リンク切れるかも。

配線には、アセンブリー品のメス~メス 10cmのこれ使います。
メス~メス、メス~オス、オス~オスが夫々40本ずつ、全部で120本!
これでお値段 US$ 1.84+送料0.42
その内、ピンとカシメ具も欲しいなあ~

思ったより配線のスペースがいるんですよね~
2叉になるVCCとGND以外は10cmのまま使ったのでギュウギュウです。
上側から
組み上がりました!
中の空洞が少ないので、約150g 軽過ぎず、いい感じの重さになりました。
USBに繋いでみます。
LEDは、赤:TX、緑:Power、右上は、RXなのでスケッチ書込時しか点灯しません。
別アングルも
CNC2418とこのコントローラーを夫々USBに接続し、
CNCの主電源なしでCandle(GRBL Controller)を起動してPC画面で確認です。
XY軸のJoysitickボタンを押すとKeyboard ControlがONになり無事動作します。
ケースに組み込んだ状態は、下の写真のブレッドボードの時から90°回転しているので
X軸とY軸が逆、X軸は左右も逆、Z軸も90°回転しています。
一応、すべての動作はOKです。
後から調整することを想定して配線を全てコネクタにしたわけでして
配線を修正して、また蓋をします。

しばらく操作していると
Joystickは、左側の方が微妙な操作ができる事がわかってきました。
こちらの方が特に左右の微妙な動作をできるのです。
親指ってのは、健が真っ直ぐの時が一番精密な動きをしてくれるのだろうか?
こちらは、親指の健が曲がるからなのか?
人差し指と親指間の筋肉が収縮しているかなのか?、わかりませんが
特に左右に動かす時に震えたりするのです。上下動作は問題ありません。
もしかしてマーティーの手がポンコツなのだろうか?
でもこれを見直して安心しました。
PS4のコントローラーのJoystickの位置が内側にあるのは同じ理由ではないだろうか?
人間工学に基づいてレイアウトが設計されているんだと思います。
ということで、XY軸用とZ軸用は、入れ替えます。
Candleの操作画面と同じ並びになるわけです。
左をXY軸用、右をZ軸用に配線変更して(ほんとコネクタ式にしといてよかった)
X軸の左右逆は、スケッチを修正して。
Y軸は、どうしようか?
ステージが手前に来る時は、座標はマイナス側に動くのですが
Joystickを手前に倒した時にステージも手前に動いた方が直感的に判り易いかな~
結局、左から右側の仕様に変更しました。
(実は、Y軸の前後、これとは逆が良いのに最後に気づくのであります^^;)
では、Candle画面と一緒の動画!
ノブのSWでStepとFeedが初期化されます。
ノブの傾きでStepが0.01 ⇒ 0.1 ⇒ 1.0 と変わります。
大きく倒すと移動量が大きくなります。
途中、斜めになったのでXY両方が変化しています。
CNC2418実機操作中の動画です。
ノブの傾きで移動量が3段階に変わっています。
一瞬倒すと単発移動、0.3sec以上倒していると連続移動になります。
なかなか使い心地良いです。
ただ、Candle画面をアクティブにしていないと
ブログの画面なんかに「7777771-----+++」とキー連打がでたりまします。

USBケーブルが短いのでカールコードの長いのにしたいなあ~
ということで、球形がてら、これをポチっておきます。
40cm~250cm  -8%Sale中 US$ 1.84!

USBには、マウス、CNC2418に、このJoystickで3ポート
これにUSBカメラなんか付けると、USB-HUB必須ですね~
CNC2418側にUSB-HUBがあると便利そうだな~

その後、実際の位置合わせ作業を実感するために、
ステージに図面を置いて、操作しております...と
Y軸の動作、スピンドルを手前に持ってきたい時は
ジョグを手前に(つまりステージは奥へ)の方が直感的なのです(-_-;)
逆なのでした! 冷静に考えれば、当たり前ですね~
ステージだけ見ていたので錯覚していたわけです。
というわけで、Y軸操作は、これに変更!
右往左往してCandle画面と同じがいいという結論になりました。
最後に組込後の最終版スケッチです。
まだ、たまにStepが小さい方に狂う問題がありますが、
ControlをON/OFFし直せば、戻るので、まあいいか^^;
それにしても、USBデバイスがこんなに手軽に作れるなんて、
ATmega32u4というか、SparkFun Pro Micro、なかなか面白いです。
もう一個ポチっておこうかな!

Arduino IDE v1.8.3
***組み込み後の最終版スケッチ***
/*
 Joystick Controller for "Candle(GRBL Contoroller)"
 with SparkFun Pro Micro, Arduino Leonardo (ATmega32u4)
 A0:Joystick X軸のADC
 A1:Joystick Y軸のADC
 A2:Joystick Z軸のADC
 D2:Scroll Lock キー用(LOWアクティブ、内蔵Pull-UP)
 D5:Keyboard Control Enable LED(ボード内のTX LED)
 ケースに組み込み完成バージョン
 ProMicro_Joystick_FIX_11.ino 2018.6.13 Ver.1.1 Marty Since 2018.5.20 
*/

#include "Keyboard.h"
// Keyboard.write() では、USB-HIDのKeyCode+136しないと、ほとんどの場合で動作せず。
#define KEY_ScrLk 71+136  //「ScrollLock」キー(USB-HID)
#define KEY_4 33+136  //「4」キー(X-Axis Right)
#define KEY_6 35+136  //「6」キー(X-Axis Left)
#define KEY_2 31+136  //「2」キー(Y-Axis Front)
#define KEY_8 37+136  //「8」キー(Y-Axis Back)
#define KEY_3 32+136  //「3」キー(Z-Axis Down)
#define KEY_9 38+136  //「9」キー(Z-Axis UP)
#define KEY_1 30+136  //「1」キー(Step Down)
#define KEY_7 36+136  //「7」キー(Step UP)
#define KEY_Minus 86+136  //「-」キー(Feed Down)
#define KEY_Plus 87+136  //「+」キー(Feed UP)
#define KEY_0 29+136  //「0」キー(Spindle ON/OFF)(今回未使用)
#define KEY_5 34+136  //「5」キー(Stop)(今回未使用)
#define KEY_Slash 84+136  //「/」キー(Spindle Speed Down)(今回未使用)
#define KEY_Asterisk 85+136  //「*」キー(Spindle Speed UP)(今回未使用)

const int xAxisPin = A0;      // joystick X-Axis
const int yAxisPin = A1;      // joystick Y-Axis
const int zAxisPin = A2;      // joystick Z-Axis
const int KeyContEnablePin = 2;    // for Keyboard Control ON/OFF
const int ledPin = 5;      // Keyboard control LED (TX LED on board ProMicro)
const short xAxis = 0;
const short yAxis = 1;
const short zAxis = 2;

boolean KeyContIsActive = false;   // Keyboard Control Enable/Disable
short lastSwitchState = HIGH;      // 直前のControl state (LOW Active)
// STEP        :Continuous、0.01、0.1、1、5、10、100
// StepPosition:     1    、  2 、 3 、4、5、 6、 7 
short StepPosition = 2;            // 現在のStep Position
short LastStepPosition = 2;         // 直前のStep Position

short xyzRead[3] = { 511, 511, 511 };     // XYZ軸のADC読取値 (初期値 1023÷2)
short xyzCenter[3] = { 511, 511, 511 };   // XYZ軸の実測センター平均値 (初期値 1023÷2)
short xyzCorrectCenter[3] = { 0, 0, 0 };  // センター補正値(Joystick VRバラツキ補正)
short xyzJoyPosition[3] = { 4, 4, 4 };  // JoystickのPosition(7段階) 4:中央
short last_xyzJoyPosition[3] = { 4, 4, 4 };// 直前のJoystick Position
short PosThreshold[3] = { 80, 40, 170 }; // 検出する最低時間の閾値(約msec)ノブ傾きにより3段階
unsigned long xyzPosCont[3] = { 0, 0, 0};// 同じPositionに停滞している経過時間をカウント
short xyzWaitCont[3] = { 0, 0, 0 };     // Single Step移動時の無効化時間カウント
short ContinuousMode = 0; // 連続移動モード 0:OFF、1:ON(0:シングル移動モード)

void setup() {
  pinMode(KeyContEnablePin, INPUT_PULLUP);  // Keyboard Control Enable SW Pin
  pinMode(ledPin, OUTPUT);          // LED pin
  Keyboard.begin();
}

void loop() {
  CheckKeyContSW_Init();  // Check Keyboard Control ON/OFF Switch と Initialize
  if (KeyContIsActive) {  // Keyboard Control ONであればJoystick Move処理する
    JoystickControllMain();  // CandleをKeyboard ControlにしてJoystickで操作する処理
  }
    delay(7);  // analogReadを各軸10回平均で3msec、loop total time 約10msecにする
}


void CheckKeyContSW_Init() {  // CheckKey & Initialize
  short switchState = digitalRead(KeyContEnablePin);  //Keyboard Control EnableキーでControl ON/OFF
  // switchStateは、変化したか?
  if (switchState != lastSwitchState) {
    // switchStateがLOWならば、Keyboard control stateをトグルする
    // 「ScrollLock」キー送出してCandleをKeyboard Controlに切り替える
    if (switchState == LOW) {
      KeyContIsActive = !KeyContIsActive;
      // 以下は、switchStateが変化した直後のみに1回だけ行う為、この位置にある
      Keyboard.write(KEY_ScrLk);  //ScrollLockキーでCandleをKeyboard Controlモードへ(トグル動作)
      // turn on LED to indicate Keyboard control state:
      digitalWrite(ledPin, KeyContIsActive);
      if (KeyContIsActive) {
        InitStepFeed();  // Step、Feedの初期化
        delay(500);  // 0.5sec delayで安定させ、XYZセンター値を取得し補正値を保存
        InitCorrectCenterXYZ();  // XYZ軸センター値の補正
      }
    }
  }
  // save switch state for next comparison:
  lastSwitchState = switchState;
}



void InitStepFeed() {
//  StepとFeedの初期位置設定
// 「1」6回、「7」1回で、Step:0.01 に初期値を設定
// 「-」5回、「+」3回で、Feed:500 に固定
  // Step初期値 0.01に設定
  for ( short i=1; i<=6; i++ ) {
    Keyboard.write(KEY_7); // 「7」6回 StepPosition最下位に設定、('7')では、まれにダメなことがある
  }
  Keyboard.write(KEY_1);  // 「1」1回、('1')では、まれにダメなことがある
  StepPosition = 2; // StepPosition:2(0.01)に設定
  LastStepPosition = 2;
  // Feed初期値 500に設定
  for ( short i=1; i<=5; i++ ) {
    Keyboard.write(KEY_Minus);// 「-」5回、('-')や(86)ではダメ
  }
  for ( short i=1; i<=3; i++ ) {
    Keyboard.write(KEY_Plus);// 「+」3回、('+')や(87)ではダメ
  }
}


void InitCorrectCenterXYZ() {
  short xyzReadSum[4] = { 0, 0, 0 };  //XYZ軸のセンター値を10回読み込み積算する
  for (short i=0; i<10; i++ ) {
    xyzRead[xAxis] = analogRead(xAxisPin);
    xyzRead[yAxis] = analogRead(yAxisPin);
    xyzRead[zAxis] = analogRead(zAxisPin);
    xyzReadSum[xAxis] += xyzRead[xAxis];
    xyzReadSum[yAxis] += xyzRead[yAxis];
    xyzReadSum[zAxis] += xyzRead[zAxis];
    delay(10);
  }
  for (short Ax=0; Ax<3; Ax++ ) {
    xyzCenter[Ax] = xyzReadSum[Ax] / 10;  // XYZ軸センター実測10回の平均
    xyzCorrectCenter[Ax] = xyzCenter[Ax] - 511;  // XYZ軸センター補正(Parts VRバラツキ補正)
  }
}


void JoystickControllMain() {
    // 各軸ADC10回の平均を取る Convertion Total Time ≒3msec  
    short xReadSum = 0;
    short yReadSum = 0;
    short zReadSum = 0;
    for ( int i=0; i<10; i++ ) {
      xReadSum += analogRead(xAxisPin);  // Convertion Time = 100μsec
      yReadSum += analogRead(yAxisPin);
      zReadSum += analogRead(zAxisPin);
    }
      xyzRead[xAxis] = xReadSum / 10;
      xyzRead[yAxis] = yReadSum / 10;
      xyzRead[zAxis] = zReadSum / 10;

      Joy_Pos_Detector(xAxis);  // Joystickの傾き検出
      if (xyzJoyPosition[xAxis] < 19) {  // Joystick Center以外の時
        MoveProcess(xAxis);   // JoystickでのX軸の移動処理
      }
      Joy_Pos_Detector(yAxis);
      if (xyzJoyPosition[yAxis] < 19) {
        MoveProcess(yAxis);   // JoystickでのY軸の移動処理
      }
      Joy_Pos_Detector(zAxis);
      if (xyzJoyPosition[zAxis] < 19) {
        MoveProcess(zAxis);   // JoystickでのZ軸の移動処理
      }
}


void Joy_Pos_Detector(short Axis) {
//  1~ 6 X-Axis Right~ Left 1.0 ⇒ 0.1 ⇒ 0.01 ⇒ 0.01 ⇒ 0.1 ⇒ 1.0
//  7~12 Y-Axis Front~ Back 1.0 ⇒ 0.1 ⇒ 0.01 ⇒ 0.01 ⇒ 0.1 ⇒ 1.0 
//  8~18 Z-Axis Down ~ Up   1.0 ⇒ 0.1 ⇒ 0.01 ⇒ 0.01 ⇒ 0.1 ⇒ 1.0
// 19~20 Center X、Y、Z
//
// STEP        :Continuous、0.01、0.1、1、5、10、100
// StepPosition:     1    、  2 、 3 、4、5、 6、 7 
//
  // 0 以上 ~ 20未満:Frot/Left/Down 1.0mm Step
  if (( 0 <= xyzRead[Axis] ) && ( xyzRead[Axis] < 20)) {
    StepPosition = 4;
    xyzJoyPosition[Axis] = 1 + Axis * 6; // X:1, Y:7, Z:13
  }
  // 20 以上 ~ 312+xyzCorrectCenter[Axis] 未満:Frot/Left/Down 0.1mm Step
  if (( 20 <= xyzRead[Axis] ) && ( xyzRead[Axis] < (312 + xyzCorrectCenter[Axis]))) {
    StepPosition = 3;
    xyzJoyPosition[Axis] = 2 + Axis * 6; // X:2, Y:8, Z:14
  }
  // 312 以上 ~ 507+xyzCorrectCenter[Axis] 未満:Frot/Left/Down 0.01mm Step
  if (((312 + xyzCorrectCenter[Axis]) <= xyzRead[Axis] ) && ( xyzRead[Axis] < (507 + xyzCorrectCenter[Axis]))) {
    StepPosition = 2;
    xyzJoyPosition[Axis] = 3 + Axis * 6; // X:3, Y:9, Z:15
  }
  // 507+xyzCorrectCenter[Axis] 以上 ~ 515+xyzCorrectCenter[Axis]未満:中央の不感帯
  if (((507 + xyzCorrectCenter[Axis]) <= xyzRead[Axis] ) && ( xyzRead[Axis] < (515 + xyzCorrectCenter[Axis]))) {
    xyzJoyPosition[Axis] = 19 + Axis;  // センターは、最も大きい番号にする(case処理で最後にするため)
    last_xyzJoyPosition[Axis] = xyzJoyPosition[Axis];
  }
  // 515 + xyzCorrectCenter[Axis] 以上 ~ 710 + xyzCorrectCenter[Axis] 未満:Back/Right/Up 0.01mm Step
    if (((515 + xyzCorrectCenter[Axis]) <= xyzRead[Axis] ) && ( xyzRead[Axis] < (710 + xyzCorrectCenter[Axis]))) {
    StepPosition = 2;
    xyzJoyPosition[Axis] = 4 + Axis * 6; // X:4, Y:10, Z:16
  }
  // 710 + xyzCorrectCenter[Axis] 以上 ~ 1000未満:Back/Right/Up 0.1mm Step
  if (((710 + xyzCorrectCenter[Axis]) <= xyzRead[Axis] ) && ( xyzRead[Axis] < 1000 )) {
    StepPosition = 3;
    xyzJoyPosition[Axis] = 5 + Axis * 6; // X:5, Y:11, Z:17
  }
  // 1000 以上 ~ 1023以下:Back/Right/Up 1.0mm Step
  if (( 1000 <= xyzRead[Axis] ) && ( xyzRead[Axis] <= 1023 )) {
    StepPosition = 4;
    xyzJoyPosition[Axis] = 6 + Axis * 6; // X:6, Y:12, Z:18
  }
  //異常値
  if (( 1023 < xyzRead[Axis] ) || ( xyzRead[Axis] < 0 )) {
    // Serial.println( "ERROR" );
  }
}


// 応答時間&不感帯判定、シングル/連続モード判定
void MoveProcess(short Axis) {
  if ( last_xyzJoyPosition[Axis] == xyzJoyPosition[Axis] ) {
     xyzPosCont[Axis] = xyzPosCont[Axis] + 10;
     // 同一Positionに最低閾値~300msec定位していればシングルstep移動命令
       if ( PosThreshold[ StepPosition - 2 ] <= xyzPosCont[Axis] && xyzPosCont[Axis] < 300 ) {  
          if ( xyzWaitCont[Axis] == 0 ) {
          ContinuousMode = 0;  // 連続モードOFF
          Send_Key_Comand(Axis); // Joystick Positionに応じてコントロールするキー操作
          xyzWaitCont[Axis] = xyzWaitCont[Axis] + 10;
        } else {
          // 一定期間(300msec弱)不感時間を設ける為のカウンタ
          xyzWaitCont[Axis] = xyzWaitCont[Axis] + 10;
        if( xyzWaitCont[Axis] > 290) {
             xyzWaitCont[Axis] = 0;
          }
        }
      }
      // 同一Positionに300msec以上定位していれば連続移動命令
      if ( xyzPosCont[Axis] >= 300 ){
        // 一定間隔で出力する連続モード
        ContinuousMode = 1;  // このモードでは、Step設定不要なので、連続モードON
        Send_Key_Comand(Axis);  // Joystick Positionに応じてコントロールするキー操作  
        delay(150);
      }
  } else {
    // 初めてのPositionにした時の初期化処理
    // ノイズも含め全ての検出は、最初にここを通る
    last_xyzJoyPosition[Axis] = xyzJoyPosition[Axis];
    xyzPosCont[Axis] = 0;
    xyzWaitCont[Axis] = 0;  
  }
}


void Send_Key_Comand(short Axis) {
// Joystick位置に応じてStepを設定
  // STEP        :Continuous、0.01、0.1、1、5、10、100
  // StepPosition:     1    、  2 、 3 、4、5、 6、 7 
  short StepDiff = StepPosition - LastStepPosition;  // 前回Stepから今回Stepへの移動数(キー打数)
LastStepPosition = StepPosition;
if ( ContinuousMode == 0 ) {  // シングル移動モードの時だけ毎回Step設定が必要、連続モードでは不要
// Step設定
if ( StepDiff > 0 ) {
for ( short i=0; i< StepDiff; i++ ) {
Keyboard.write(KEY_1); // Step上げる「1」キー(ここも'1'ではダメ)
}
}
if ( StepDiff < 0 ) {
for ( short i=0; i< -StepDiff; i++ ) {
Keyboard.write(KEY_7); // Step下げる「7」キー(ここも'7'ではダメ)
}
}
}

  // switchの中( (xyzJoyPosition[Axis] -1)/ 3 + 1 )だとcaseを少なくできる
  short MoveDirection = (xyzJoyPosition[Axis] -1)/ 3 + 1; // 移動方向
  // X-RIGHT, X-LEFT, Y-FRONT, Y-BACK, Z-DOWN, Z-UP, XYZ-CENTER
  //    1   ,   2   ,   3    ,   4   ,   5   ,  6  ,    7
  switch ( MoveDirection ) {  // Joystick Positionに応じてキー押下を発行
    case 1:  // X RIGHT xyzJoyPosition[Axis]=1 or 2 or 3
      Keyboard.write(KEY_4);  //「4」
    break;
    case 2:  // X LEFT xyzJoyPosition[Axis]=4 or 5 or 6
      Keyboard.write(KEY_6);  //「6」
    break;
    case 3:  // Y BACK xyzJoyPosition[Axis]=7 or 8 or 9
      Keyboard.write(KEY_2);  //「2」
    break;
    case 4:  // Y FRONT xyzJoyPosition[Axis]=10 or 11 or 12
      Keyboard.write(KEY_8);  //「8」
    break;
    case 5:  // Z DOWN xyzJoyPosition[Axis]=13 or 14 or 15
      Keyboard.write(KEY_3);  //「3」
    break;
    case 6:  // Z UP xyzJoyPosition[Axis]=16 or 17 or 18
      Keyboard.write(KEY_9);  //「9」
    break;
    case 7:  // センター 何もしない  xyzJoyPosition[Axis]=19 or 20 or 21
    break;
    default:  // 削除してもいいかも
    break;    
  }
}
*****

2018年6月9日土曜日

小型工作機CNC2418 その53(JoystickコントロールBOX製作)

前回、GRBL ControllerのCandleのJoystick Controllerのスケッチが完成したので
その筐体を造ります。

Joystick!中心が分かり難く、3D Model作るの複雑そうで面倒だなあ~
と探してみるとここにありました!
但し、User登録しないとDownloadできません。
Licenseには、非商用のみ許可する旨が書いてありますので
登録してDownloadしました。
色んなファイル形式で入っていますが、
Fusion360で読み込めるのは、STEPファイルだけでした。
Arduino Pro MicroもあったのでDownloadします。

Fusion360にJoystick 2個とSparkFun Pro Microを挿入して
どんな形状にしようかな~
と思案していると、こんな空きボトルが転がっておりました。
いい感じに丸みを帯びて持ちやすいかも?
ちなみにマーティーは髪が薄いのでこんなもの必要ありません^^;
厚みは、約40mmで手のひらに当たる部分が丸いので
持ち易いですが、根元の方がちょっと
反対にしてみます。
おっ!根本が細いと、何となくしっくりする感じです。
丁度Joystickが2個並びそうな幅だし、
これを基本形状にしてモデルを描いていくことにします。
それと、これは、忘れてはいけない一品です。必ず参考にすべきでしょう!
流石に操作しやすいPS4のコントローラー!
ボタンの所にもう一つのJoystick置いても丁度よいです。
Joystickは、真横よりも、親指の付根を中心にした円弧状にあった方が良いみたいです。
さんざんこだわりと見せておいて、できたのは、安易なものに^^;
両面切削で曲線美を出そうかとも思ったのですが、
どうも集中力悪く、上下は間違えないのに、左右を間違えること4~5回(-_-;)
ようやくできたのがこれ!
裏返して正確な位置決めする手段をまだ考案してないので片面加工だけなのです。
角のフィレットは、手加工を予定しております(-_-;)
透明モデル
アングル変えて
シェーバーって感じもしてきました。
真上から。
右利き専用です。薄い所は配線用の空洞です。
実は最初は、分厚いので木材で作ろうかと
下から蓋をするだけの構造にしていましたが
以前使ったポリエチレンの抗菌まな板(10mm厚)が残っているのを思い出し
急遽、3枚構成に分解したのです。
加工するために底面を合わせて横に並べます。
よーく見直すと弱そうな所や逃げ穴忘れがあるので修正していきます。
パーツはこの様に取り付きます。
Joystickの基板がいつの間にか円形にカットされちゃってます(-_-;)
ひっくり返すとこんな感じ
この様に組み立てます。
実際は白なのでこんな感じかな。
最後に手加工で丸みを帯びさせる予定です。
真ん中の穴は、鼻と口ではありません、SparkFun Pro MicroのLED用です。
何だか、とぼけたロボットの顔みたいにも見えてきた...
ポリエチレンの抗菌まな板を3ブロック同時に加工したいので
CNC2418のステージに収まるように配置を考えます。
赤枠がステージの外形です。ギリギリで入りました。
左下の円は、前に加工してくり抜いた部分、縦線は、CNC2418のステージの溝部です。
では、Fusion360 CAMに行きます。
設定は、この時の表のPE(ポリエチレン)を使います。
設定表:<PVC・アクリル・ABS・ポリエチレン>
 エンドミルタイプ:スパイラル・タイプ(Flute:1or2)必須

D[mm]0.60.81.01.21.52.02.5 3.175
 切削送り速度 
Vf
[mm/min] 
 2000rpm PVC 100 130 160 180 200 230 240240
 3000rpm アクリル
ABS
150190230270300340360360
 4000rpm PE190260310350410460480480
深さ[mm] 0.3 0.4 0.4 0.5 0.6 0.6 0.70.7

<1>各ブロックを合体するためのビス穴とLED用の穴から
[2D]-[円形]でΦ2.0mm 1-Flute PVC用スパイラル エンドミル
深さ:0.6mmずつです。
<2>TOPブロックの木ねじの先端をねじ込む穴。
ここは、[ドリル]で[深穴ドリル-完全退避]を使います。
0.5mm掘る毎に上がってきて屑を出します。
溶着防止のためです。
CAMの設定では、ドリルビットを使いますが
実際には、Φ2.0mm 1-Flute PVC用スパイラル エンドミルをそのまま使います。
<3>BOTTOMブロックの基板足避けの穴
[3D]-[ポケット除去]
Φ2.0mm 1-Flute PVC用スパイラル エンドミル
深さ:0.6mmずつです。
<4>BOTTOMブロックの輪郭段差
[2D]-[ポケット]
3.175mm 2-Flute PVC用スパイラル エンドミル
深さ:0.7mmずつです。
<5>TOPブロックのJoystickノブ穴、下の方は球形傾斜で、穴径が小さくなります。
[3D]-[ポケット除去]
3.175mm 2-Flute PVC用スパイラル エンドミル
深さ:0.7mmずつ、傾斜部は、球面状なので深さ:0.1mmずつです。
<6>MIDブロックの中面取り
[2D]-[ポケット]
3.175mm 2-Flute PVC用スパイラル エンドミル、深さ:0.7mmずつです。
全体を3.0mm 面取りします。
<7>MIDブロックの深い堀り
[3D]-[ポケット除去]
3.175mm 2-Flute PVC用スパイラル エンドミル
深さ:0.7mmずつでパーツが入る部分を掘ります。
<8>MIDブロックの細かいボスなど
[3D]-[ポケット除去]
1.5mm 2-Flute PVC用スパイラル エンドミル
深さ:0.5mmずつで、3.175mmで切削できなかった所をやります。
<9>MIDブロックの基板が入る窪みの角の掘り残しを整えます
[2D]-[輪郭]
0.8mm 2-Flute PVC用スパイラル エンドミル
深さ:0.6mmずつです。
<10>全ブロックの輪郭掘り出しの最終工程。
[2D]-[輪郭]
2.0mm 1-Flute PVC用スパイラル エンドミル、深さ:0.6mmずつ。
この部材は柔らかいので、高めのタブを付けます。
全部通しでシミュレーション!
ん~ん、ぶっ通しで3:30:54とでていますね~
Fusion360の自動バージョンアップで
「G28 Safe retracts」はDefaultの「はい」に戻っているので
ポスト処理する前に必ずここを確認して「いいえ」にします。
これを忘れると、何故か、最初に下がりながら移動して不意な切削筋が入るのです。
全10行程です。
エンドミルは、2.0 ⇒ 3.175 ⇒ 1.5 ⇒ 0.8 ⇒ 2.0mmの交換です。
同じエンドミルは、合体してポスト処理してもいいのですが
工程ごとに確認しながらやりたいので、個別にポスト処理します。
工具交換回数を減らしたいので順番を変えて
最初の2.0mmエンドミルの加工を後ろに持っていくことにして
3.175 ⇒ 1.5 ⇒ 0.8 ⇒ 2.0mmの交換にします。

さーていよいよ切削ですが、
久しぶりなのでモリブデングリスを塗ります。
部材は、銀イオン配合の抗菌まな板、ポリエチレン 320 x 200 x 10mmです。
食品衛生法にも適合しています。
3ブロックをレイアウト印刷した紙をまな板の上において位置決めします。
セット完了!
手前の丸穴が、以前使った部分です。
GRBL ControllerのCandle v1.1.7を立ち上げて切削開始です。
ちょっと無理している様な音なのでSpendle Speed 120%に上げます。
工程<4>約9分で完了。
工程<5>24分
工程<6>32分
これは、アクリルに比べてちょっと粘りがあるようですが、溶着は起きにくいようです。
Spindle 130%、Feed 110%にしています。
GRBL v1.1fのOverridingは、便利です。
1-Fluteは、面取りでは、筋が多くてキレイにならないですね~
2-Fluteでやるか、もう一回、方向を変えて切削するのがいいかも?
支障はないのでこのままです。
あっ!位置合わせの失敗です。
赤枠の所がギリギリ、ここは3mmないといけなかったのですが
もう少し部材全体を手前にセットしないといけませんでした。
続いて最も複雑な所、工程<7>1時間もかかりました。
ステッピングモーター、ドライバがかなり熱くなっているので休憩です。
んっ?何だか浅いぞ~
3mmの深さのはずの所が2mmもありません?
もう一度Fusion360 CAMを確認しますが問題ないのです。
切削中にCandleの画面を見た時もちゃんとZ軸-10mmまで掘っていたし
まさかと思いCandleで「$$」して設定を確認してみると

何と!!! $102=495.000になっています。Z軸のstep/mmの設定です。
この時にXYZ較正して、$102=793.388  にしたはずなのです。
いつ変わったのだろう?しかも、なぜこんな変な値なんだろう?
実験するとしても半分近い値にするハズもないし、不思議でなりません。
そういえば、「495」の数字、前回のJoystickのスケッチ作成中に使いました。
CNC2418と接続してCandleでスケッチの動作確認をしている時に
Arduino IDEのシリアルを立ち上げていなかったので
Candleの方に数値が吐き出され、この数値が入り込んのではないかと考えらます。
でも$102=とセットじゃないと設定されないはずですが
Keyborad.write命令と合わさって偶然書き換わったのでしょうか?
気をつけないといけません。
さて気を取り直して、$102を設定し直します。
他の設定は、大丈夫でした。
幸い浅めだったので同じ所を再度掘り直します。
ついでに2-Flute PVC用に変えます。
同じGcodeを使ったので、途中までFeed200%で早送りして進めます。
手前の最初の深さ3mm溝。今度はOKです。
次の穴もちゃんと下まで貫通しました。
最初の時にこの穴で浅いのを気づくべきですが、なぜか...(-_-;)
Candle画面の深さが-10mmだったので安心していたのです。
次の工程の切削状況を動画で!
3.175mm 2-Flute スパイラル・エンドミル
深さ-0.7mm Spindle:5200rpm、Feed:530mm/min
エンドミルに白く溶着しているように見えますが
これは、部材のクズが糸状になって巻き付いているのです。
時々、集塵機で吸い取られてなくなるのがわかります。
この動画は、最後の貫通時の様子です。
最後の方は、深さ0.7mmずつ切削した余りで0.1~0.2mm程に薄くなっています。
部材のポリエチレンは、柔らかいので薄くなるとこんな感じです。
まさに厚めのポリエチレン袋をちぎった感じなのです。
ここまで終わった瞬間にUSBケーブルに当たってしまいOfflineに!
あ~あっ!座標が!行方不明になる~~っ _| ̄|○
幸いCandle上に最後の位置が表示されていたので、何とか原点に復帰できました。
3.175mmで掘り残したボス部を1.5mmでやります。
ステージに固定した状態でJoystickを入れてみます。
クリアランス0.2mmにしていたので、ギチギチのピッタリです。
角を0.8mmエンドミルでの仕上げるつもりでしたが不要のようです。
この後、左端から、右端の原点に復帰中にX軸が途中で停止しました!
Y軸は動いているのにです。
過去にも1時間コースの切削やりましたが、止まるのは初めてです。
熱の性かも? 特にステッピングモータードライバは、触れられないくらいです。
その内、何とかせねば...
今度は、座標が完全にパア_| ̄|○
今日は、なぜか災難続き???原点の再設定に一苦労です。

2.0mmエンドミルに交換して、3ブロックを取付用のビス穴です。
ランプで回りながら降りてくる切削モードになっていたので
15個で21分もかかってしまいました。
最後は、外形をタブ付きで掘り出しです。45分。
全行程完了です。
朝から始めたのに夜になっています。
全体風景。
左のは、1年程前に作ったサイクロン集塵機、ホースはFREE!手持ちが便利なのです。
右のは、ここで作った緊急操作ボタンボックス。
ステージから取り出しました。切削面。
裏面
タブをカットして取り出します。
どんどん顔に見えてきます。
家族からも「骸骨を造っているのか」と言われてしまいました。
左端は、「ムンクが叫び終わった後」だとか(笑)
それにしてもこの部材は、溶着しにくい代わりにやたらと屑ヒゲが多いです。
まあ、ぴったりの嵌合には驚いてもらえました。
でパーツを組み込もうと思ったら
あっ!穴あけ忘れ発覚! 途中で2回も原点喪失したので焦ったようです。
このブロックだけ原点を変えてポスト処理し直します。
これもランプで回りながら降りてくる切削モードになっていたので、19分も。
幸い、パーツの足を逃げるためなので穴位置は多少ズレても大丈夫です。
やっと、JoystickとSparkFun Pro Microを装着!
Joystickの5ピンコネクタ位置が僅かにずれて当たっているのと、
ProMicroが、クリアランスの内側を切削していましたので入らず^^;
少ーしカッターで修正。柔らかくてカッターでの加工はやり易いです。
きっちり入りました!
ひっくり返して
きっちりピタッとハマっているので、持ち上げても落ちません。
下ブロックの上に乗せます。
本来なら下側に位置決めボスをつけてパーツを乗せる構造がよかったですね~
当初、下側は、単純な蓋の構想で始めたのでこんな構造になったわけです。と言い訳。
ここは、外周壁で入れ子構造にしています。
のでピッタリ!
クリアランス0.2mmで無理なく入るしガタはありません。
QFPGアダプタソケットの時の経験が活きております。
あの時は、クリアランス 0.1mmでかなりギチギチだったのです。
上ブロックは、片側だけ加工なので、入れ子にできず、かぶせるだけです。
Joystickノブの傘部が入り込むので穴は球形状になっています。
仮組み終了!
いい感じです! 悪くはないでしょう!
下ブロックの裏側から皿頭の木ねじで3ブロックを固定します。
皿ネジの頭部分を凹ますためザグリ加工します。
これもさほど精度を要しないので
原点穴と遠くの穴の2点の座標が合うように固定して加工します。
こんな感じで木ねじが入ります。
さて、操作感のチェックです。
薬指の角を手加工でフィレットつければ、なかなか良さそうです。
Joystickノブとケースのスレは、全くなく、スムーズな動きです。GOODです!
参考にさせて頂いた3D Modelがバッチリだったお陰ですm(_ _)m
自作していたらズレていた可能性大です^^;
下の方からUSBケーブルがでてくる予定です。
上の右スティックの方が、微妙なXY操作がやり易そうなので
左側は、Z軸用にしよう!
Candleの画面とは逆になるのでちと心配ですが。
数々の失敗^^;とトラブルがあり想定外の時間と気力を使い果たし
配線、実働確認にまでこぎつけませんでした~
改めてメカ屋さんの凄さを実感です!
一度、この様な設計の基本を勉強した方がよさそうです。
TO BE CONTENUED~