Final Project
Here is where I will be documenting the process of making my final project. As this is my first full integration project, I wanted to make something simple. What I decided to make was an RGB Macropad, a convenient place with all the keybinds and media keys I need. I will be incorporating laser cutting, 3D printing and microcontrollers into a fully integrated project. The raw files can be aquired below and the rest of this page is my week to week documentation of the process.
Table of contents
List of Materials
Raw Files
Week 1
Week 2
Week 3
Resources Used
List of Materials
Parts to purchase | Quantity |
---|---|
Arduino Pro Micro | 1 |
4x4 Membrane Keypad | 1 |
WS2812 Neopixel | 6 |
Frosted Acrylic Sheet | 1 |
Parts to 3D print | Quantity |
Center | 1 |
Mount | 1 |
Corner Spacer | 4 |
Feet | 4 |
Parts to laser cut | Quantity |
Top Plate | 1 |
Keypad Support | 1 |
Front Plate | 1 |
Back Plate | 1 |
Bottom Plate | 1 |
Side Plate | 2 |
Raw Files
[Google Drive folder with the raw files.]Week 1
Ideation
Originally, I wanted to make a mechanical tenkey pad for macros and shortcuts from scratch, soldering my own mechanical switches together in a switch matrix. However, because of the lack of time, I decided to use a 4x4 Matrix Membrane Keypad, which achieves the same purpose. With the main component decided, I went ahead and sketched out a basic case design. I took reference from https://create.arduino.cc/projecthub/brian-lough/the-simplest-diy-macro-keypad-d2e428 in order to learn more about the 4x4 matrix keypad.
Sketch
data:image/s3,"s3://crabby-images/829da/829da3f5485e6f535befc0ecb3293c8e1533d751" alt=""
The case consists of 4 acrylic walls, connected at the corners with 3d printed pieces, and sandwiched between 2 acrylic plates. The keypad rests on platforms on the corner pieces and is held down by the top plate. I plan to use M3x25 screws to connect everything together. With this sketch, I started to 3D model the case
3D modelling
Version 1
The differences between the sketch and this first version is the addition of the mount for the microcontroller and that the feet has not been modeled yet. However, after modelling it I realised that I did not have any place to mount the neopixel strips I planned to use.
Version 2
Changes were made to the corner pieces such that now there is a "floating" panel in the middle where the neopixels can stick onto. The keypad will still rest on the top of it. Originally, the turqoise and pink parts (pictured below) were a single body, but I separated them so that there will be no supports during 3D printing.
Another change made was adding more space for the keypad. The keypad is very thin and floppy on its own so I plan to stick it onto a laser cut panel in order to give it more rigitidy and strength. The thickness of the acrylic plate was accounted for in the space between the top plate and the place where it will rest.
data:image/s3,"s3://crabby-images/6ffbf/6ffbf293884f8e392d7e3bb5b4790ca220005bff" alt=""
3D printing and laser cutting
Machine and settings used for 3D printing
Machine | Ultimaker 2+ |
Material | Orange PLA |
Layer Height | 0.2mm |
Infill Density and Pattern | 20% Grid |
Printing speed | 80mm/s |
Supports | None |
Build Plate Adhesion | Brim |
data:image/s3,"s3://crabby-images/d76d9/d76d95abed5f0268f69826a822e9f4f8f79c12c5" alt=""
data:image/s3,"s3://crabby-images/60124/6012407c76abea59b129fee74d691e6b9455c0b3" alt=""
data:image/s3,"s3://crabby-images/bbb8d/bbb8d8afe2f388ffa04d1209eb21f39bd9299bbb" alt=""
Machine and settings used for Lasercutting.
Machine | Epilog Fusion Pro |
Material Thickness and Type | 3mm Frosted Acrylic |
Cut Settings | Speed - 18 |
Power - 90 | |
Frequency - 90 | |
Engrave Settings | None |
data:image/s3,"s3://crabby-images/63921/639214497c088cddf3cd5929300c082a8d00d401" alt=""
data:image/s3,"s3://crabby-images/277af/277af1c77272eb5e5e63f1f7f42ccba7e58ac72e" alt=""
Issues
After I got all the parts, I faced 3 issues.
Issue 1: I did not properly measure how wide the neopixel strips were before printing. The current width on the 3D model was 5mm, but the neopixels are 9~10mm wide. With the current model, only half of the width would be secured.
Issue 2: Using the brim was a mistake. Because the parts being printed were totally flat with relatively large surface area, I could have gotten away with printing without the brim. However, I still did, which resulted in lots of time being spent cleaning up the print, as the brim got into all the slots for the acrylic plate.
Issue 3: I did not account for tolerances and shrinkage during the print. As a result, the acrylic plates could not fit into their slots and lots of post processing work had to be done to make sure they fit. Moreover, the holes I modelled to be M3 ended up shrinking as well, requiring me to drill it out again using a 3mm drill.
Even with all these issues, I still sanded all the parts to make it fit in order to see how everything went together.
Week 1 assembly
data:image/s3,"s3://crabby-images/79ec9/79ec9c314594e4d660239e9263eb43f538f28809" alt=""
data:image/s3,"s3://crabby-images/140cc/140cc9a0f3ade1edb9a6b982b02f22a33327a444" alt=""
data:image/s3,"s3://crabby-images/1f6ad/1f6adddabf3f33a0d45c7ba62da11ffb5794b715" alt=""
data:image/s3,"s3://crabby-images/0bcd4/0bcd4e9db6398b22b4357fe32981f14b44012d5f" alt=""
Next week, I plan to reprint the corner piece with the correct width for the neopixel, print out the mount for the microcontroller, as well as solder the keypad and neopixels to the microcontroller.
Week 2
From the first week, I just kept the laser cut parts as there were no changes done to the design. During this week, I plan to print out the remaining parts such as the arduino mount and the feet, remodel and reprint the corner pieces as well as finish soldering the electronics.
3D Modeling
As stated in week 1, the original mounting place for the neopixel LED strips was too narrow. I extended the extrude downwards by an extra 5mm, and modelled the feet for the macropad. With this, the design is finalised.
data:image/s3,"s3://crabby-images/049bf/049bf020efe23835441066e82af63c0a71d4edd2" alt=""
data:image/s3,"s3://crabby-images/be41e/be41e0e309694f957af01c1c9e679ba10a96251c" alt=""
3D printing
Machine and settings used for 3D printing
Machine | Ultimaker 2+ |
Material | Black PLA |
Layer Height | 0.2mm |
Infill Density and Pattern | 20% Grid |
Print Speed | 80mm/s |
Supports | None |
Build Plate Adhesion | None |
data:image/s3,"s3://crabby-images/987a2/987a21cdf2fa41f5499373e7bc70de60b591ac6d" alt=""
data:image/s3,"s3://crabby-images/1c482/1c4824919478eae552d0ba35d528850f700bb849" alt=""
data:image/s3,"s3://crabby-images/afa4c/afa4c87211c585bb7da0e768e3ae9b735822be3c" alt=""
Even though the parts that the acrylic plate meshes into was not changed, somehow this print had better tolerences than the orange print. There was no sanding required to fit the acrylic side plates in. The 3mm hole still shrunk slightly and had to be drilled out. I printed the mount without supports as it was not really an area of importance and it came out fine, though just like the other holes, I had to drill the mounting holes out.
Now that the case has been made, time to move on to electronics.
Electronics
This whole project relies on an arduino pro micro board I had lying around. For this week, I just want to finish the soldering and make sure the components work before I start coding.
Wiring diagram
data:image/s3,"s3://crabby-images/cc73c/cc73c4c2fa777b6ddb7b6621ba284e0dc5f0c818" alt=""
This was the first time I was soldering so at first, the solder job was not that good, but I gained a lot of experience during the whole process.
First was soldering the connections to the keypad. To make connecting to the keypad easier, wires were first soldered to 8 connected pin headers. Afterwards the pin headers can just plug into the keypad. These connections were the first ever soldering job I have done, and its clearly visible how shoddy it is. Nevertheless, instead of resoldering it, I just trusted that it will work and shrink-wrapped the connections, just in case there was a short between the wires. Afterwards, I soldered the other end of the wires to the arduino.
data:image/s3,"s3://crabby-images/07cb5/07cb5edd6a67e846cf8aad21e55df5b52026139f" alt=""
data:image/s3,"s3://crabby-images/95a4a/95a4a632f06f6ca5e405ab66424ee1a38f5be251" alt=""
data:image/s3,"s3://crabby-images/d6e8f/d6e8fd986b256ab39f39596e2de3bef8b31e5427" alt=""
Afterwards was soldering to the neopixel strip. This part should have been easy because the neopixel only needs 3 pins to connect (Data in, 5V and GND). However, because I want to space out the individual LEDs apart, 6 solder joints ended up becoming 24. There are a total of 6 LEDs, split into 2-1-1-2, wrapping around the center part of the corner 3D print, pictured below.
data:image/s3,"s3://crabby-images/996b1/996b17882c43d2f61badbc46751f4b072f4b3757" alt=""
Due to the large amount of connections, I had a lot of practice soldering. Because I did not get the lengths right, I had to desolder and resolder about 12 more times, which gave me even more practice. This surely has given me a lot of experience soldering, although in the future I hope to get it right on the first try.
data:image/s3,"s3://crabby-images/7b02d/7b02d20df2b74032caaa044adc610711eaf73b3e" alt=""
data:image/s3,"s3://crabby-images/6cece/6cecef906ab56d039bfa39a0b7d6f6fd465585da" alt=""
After soldering to the board, the electronics has been completed.
data:image/s3,"s3://crabby-images/9a564/9a56487e416681665ad6374482a54543c299b161" alt=""
Electronics Test
Once I got the solder done, I tested the connections using the examples in arduino to see if they still work. I used the built in strandtest from the Adafruit library to test the neopixel strips and used the code for a 4x4 keypad from [here]. From the tests, the all the solder joints work with no issues, meaning I can continue with the code in week 3.
Week 3
Now that I have finished the case and the electronics, this week I plan to finish the code and final integration.
The Code
The full ino file can be found at the top of this page. In this section I will only be explaining the main parts of the code and how I got the code. For this project, 3 arduino libraries were needed. These libraries are arduino's built in Keypad.h, Adafruit_Neopixel.h by Adafruit and HID-Project.h by Nicohood.
data:image/s3,"s3://crabby-images/bf9ab/bf9ab730ccda321cb8a558ad8c6bd0eba5817cab" alt=""
First, I used the keypad code from the test example and modified it so that the keypad pattern better reflected what the buttons are supposed to be. When I soldered it, the connections were fine but I messed up the rows and columns of the keypad. Instead of resoldering I decided to fix it in code here.
data:image/s3,"s3://crabby-images/de0bb/de0bb1b9376ba26ca9cfa8154e662519bd50dadf" alt=""
In essence, the main bulk of the code is a large switch case, with each case corresponding to a button on the keypad and doing something different. For example, below is a part of the 16 cases long switch case. When button '3' is pressed on the keypad, the program recognises it as case 3 and execute the commands for case 3, which is to tell the computer to go to the next media. If button '4' is pressed instead, it will tell the computer to press left alt then tab, switching between open applications.
data:image/s3,"s3://crabby-images/67471/67471239215ce4b2a898bc62b01a39c2974141ae" alt=""
The binds can be changed anytime by changing the code. In general, numbers and letters are Keyboard.press('_') where _ is the letter or number, modifier keys like left alt are Keyboard.press(KEY_LEFT_ALT), and media keys like pause are Consumer.write(MEDIA_PLAY_PAUSE). The switch case just has to be changed to whatever bind or macro you want to set.
Majority of the cases follow something similar except switch case 'A'. I wanted to make this button cycle through different RGB LED modes, so the switch case is different. I got the code for the RGB lighting by combining the pattern updating, rainbow and rainbowCycle code from a comment by "Grumpy Mike" on an arduino forum [here] and the code for solid colors from an arduino project by Arnov Sharma [here].
How the color switching works is that first the arduino detects if 'A' has been pressed. Once it has, it will add 1 to the pattern number.
data:image/s3,"s3://crabby-images/1d746/1d746c1c7a0c7fa4f9f862bc7e529d0d1120976f" alt=""
At the same time, a function to keep updating the pattern is repeated indefinitely. Although this does not really affect the solid color options, for the last 2 options, rainbow and rainbowCycle, it is very important for the pattern to repeatedly update.
data:image/s3,"s3://crabby-images/8c421/8c42168aef975256afad9c1fd7ca750842a4d837" alt=""
The function is its own switch case, depending on the pattern number dictated by the 'A' button. The switch cases range from off to solid colors,
data:image/s3,"s3://crabby-images/d6c22/d6c22f1605857ff19bd42bb39838c5ea70cbeef0" alt=""
to the rainbow and rainbowCycle cases. Currently there is an off state, 7 solid color states and 2 dynamic color states.
data:image/s3,"s3://crabby-images/fc306/fc3061778e304a67472269e86b9fc92d55686850" alt=""
Modifying the code.
The binds can be changed anytime by changing the code. In general, numbers and letters are Keyboard.press('_') where _ is the letter or number, modifier keys like left alt are Keyboard.press(KEY_LEFT_ALT), and media keys like pause are Consumer.write(MEDIA_PLAY_PAUSE). The switch case just has to be changed to whatever bind or macro you want to set.
To add more solid color modes to the RGB, first the intervals in line 9 need to be updated. The position of the number corresponds to the switch case the interval is for. Currently I have the interval for solid colors as '20' and the intervals for rainbow and rainbowCycle as '50'. If more colors and modes are added, this line will need to be updated to reflect the change.
data:image/s3,"s3://crabby-images/fbdcd/fbdcde78456c09d6d23dff5239b45706725159c5" alt=""
Then the number of cases in line 11 need to be changed
data:image/s3,"s3://crabby-images/d7912/d79120541a047099a10b55230e257b621fb279ae" alt=""
Finally, just add another case to the switch case. The color is determined by the 3 numbers in the strip.Colour brackets. Each number corresponds to a value of RED, GREEN and BLUE respectively, ranging from 0 to 255.
data:image/s3,"s3://crabby-images/a7b02/a7b02b8190aeb96405ef744ea994a11212013917" alt=""
Testing the Code
Below is a video test of some of the functions of the keypad. Although not all the macros are shown, the other parts of the code still work as intended.
Final Assembly
data:image/s3,"s3://crabby-images/88beb/88beb228bb983ca7814510ecf028dec58a61044d" alt=""
data:image/s3,"s3://crabby-images/13644/13644b9be0c2c9b12c66128bb5adbd5dd13b6431" alt=""
data:image/s3,"s3://crabby-images/5779d/5779dd3ec32688a29454164dc8be2f7a88e078a7" alt=""
data:image/s3,"s3://crabby-images/0226d/0226d5579811251f4b8a76a16758f1a33f68a693" alt=""
data:image/s3,"s3://crabby-images/d942c/d942cd198b4355825b9af808f3d9ad7193bc0880" alt=""
data:image/s3,"s3://crabby-images/6faf0/6faf032819d49607e4b6b9cb2028997644bdbca4" alt=""
data:image/s3,"s3://crabby-images/c3243/c3243a7c0a06037182367b43f068079bad73135b" alt=""
data:image/s3,"s3://crabby-images/eb206/eb2066dc6fb96629844cf8975cb9b461c43f7c84" alt=""
data:image/s3,"s3://crabby-images/6c3bd/6c3bdac68bea27ff4c544c75e06f4ea55be53078" alt=""
data:image/s3,"s3://crabby-images/d4f57/d4f57880ae8d2ea0dbf293230b597ca0180270f1" alt=""
data:image/s3,"s3://crabby-images/9914b/9914b418e82314da90442e409c6643edc8c46a9e" alt=""
data:image/s3,"s3://crabby-images/5e577/5e57784ff2486f1dfd384e3eb0cc899a11d39971" alt=""
data:image/s3,"s3://crabby-images/26640/266408baed0d2ec18db3ae40e8ccf97b18a9392d" alt=""
data:image/s3,"s3://crabby-images/ec264/ec26431d8d7ceae553a4b1f356a3c26b35a2e738" alt=""
data:image/s3,"s3://crabby-images/b9414/b9414092b13b9c9f1d549c5c63f66fe13cae89f8" alt=""
data:image/s3,"s3://crabby-images/df518/df518e9d446db38f1b8c6851a47a2e79491ef8b9" alt=""
data:image/s3,"s3://crabby-images/2d520/2d52025a87cb78d8a3142d95a338048e31be406e" alt=""
data:image/s3,"s3://crabby-images/62e4b/62e4b321984ad3bb783a752e4f9e43469fddd86f" alt=""
data:image/s3,"s3://crabby-images/dbc55/dbc5562b1cb705205a34da4b63cfe5959d8cc5a0" alt=""
data:image/s3,"s3://crabby-images/61e28/61e280501200038f51c4e03dfd4fb14b6ad43416" alt=""
data:image/s3,"s3://crabby-images/ed09d/ed09d1b5465e768141753eb063bacf1b2def5322" alt=""
Project Video