Spisu treści:
2025 Autor: John Day | [email protected]. Ostatnio zmodyfikowany: 2025-01-13 06:58
Jest to drewniany zegar LED w stylu analogowym. Nie wiem, dlaczego nie widziałem wcześniej żadnego z nich… mimo że typy cyfrowe są bardzo powszechne. W każdym razie, zaczynamy!
Krok 1:
Projekt zegara ze sklejki rozpoczął się jako prosty projekt startowy dla routera CNC. Przeglądałem proste projekty online i znalazłem tę lampę (obrazek powyżej). Widziałem również zegary cyfrowe, które prześwitują przez okleinę drewnianą (obrazek powyżej). Połączenie obu projektów było więc oczywistym pomysłem. Chcąc rzucić sobie wyzwanie, postanowiłem nie używać do tego projektu forniru, a jedynie kawałek drewna.
Krok 2: Projekt
Zaprojektowałem zegar w Inkscape (obrazek powyżej). Projekt jest bardzo prosty z wyboru. Zrezygnowałem z prowadzenia ścieżek dla przewodów, ponieważ w tym momencie nie byłem pewien, czy chcę okablowanie promieniowe, czy obwodowe. (W końcu zdecydowałem się na okablowanie obwodowe.) Jeden neopiksel jest umieszczany w każdym z małych okrągłych otworów, aby pokazać godzinę i godzinę z dokładnością do pięciu minut. Okrąg pośrodku zostanie poprowadzony, aby pomieścić elektronikę.
Krok 3: Obróbka CNC
Zaprojektowałem ścieżki narzędzia w MasterCAM i użyłem technoRoutera do wyfrezowania zegara ze sklejki 3/4 cala. Używam do tego kawałka 15"x15", z minimalnym marnotrawstwem. Sztuczka polega na tym, aby wyprowadzić jak najwięcej drewna bez przebijania się przez drewno. Pozostawienie 0,05"-0,1" to dobry wybór w przypadku jasnego drewna. Jeśli nie masz pewności, lepiej zostawić więcej drewna, ponieważ zawsze możesz przeszlifować drugą powierzchnię. Skończyło się na usunięciu trochę za dużo drewna z niektórych części, ale na szczęście wyniki nie są z tego powodu zbytnio cierpiące.
Uwaga dla użytkowników bez dostępu do CNC:
Ten projekt można łatwo wykonać za pomocą wiertarki. Wystarczy ustawić ogranicznik w punkcie, w którym zostawisz około 0,1 cala drewna pozostałego u podstawy. Musisz być precyzyjny, ale nie zbyt precyzyjny. W końcu idealnie nikt nie zobaczy, że wszystkie diody świecą w tym samym czasie, więc możesz ujść trochę na sucho.
Krok 4: Elektronika
Elektronika jest dość prosta. Są 24 neopiksele, dwanaście do pokazywania godzin i dwanaście do pokazywania minut z pięciominutową precyzją. Arduino pro mini kontroluje neopiksele i uzyskuje dokładny czas za pomocą modułu zegara czasu rzeczywistego DS3231 (RTC). Moduł RTC ma zapasową baterię monetową, dzięki czemu nie traci czasu nawet przy wyłączonym zasilaniu.
Materiał:
Arduino pro mini (lub dowolny inny Arduino)
DS3231 tabliczka zaciskowa
Neopiksele w poszczególnych tablicach typu breakout
Krok 5: Montaż elektroniki
Połączyłem neopiksele w ciąg, używając przewodów 2,5 dla pierwszych dwunastu diod led i czterocalowego drutu dla następnych dwunastu. Mogłem użyć nieco mniejszych długości przewodów. Po zrobieniu sznurka przetestowałem go, upewniając się, że lut stawy były dobre. Dodałem chwilowy włącznik, aby włączyć wszystkie diody, tylko po to, żeby się popisać.
Krok 6: Bieg na sucho
Po eksperymentach, włożeniu diod LED do otworów i włączeniu ich wszystkich, byłem zadowolony z wyników. Więc trochę oszlifowałem przednią powierzchnię i nałożyłem powłokę PU. Skończyło się na zeszlifowaniu płaszcza później, ale dobrym pomysłem jest pozostawienie go na sobie, jeśli nie uważasz go za nieprzyjemny pod względem estetycznym.
Krok 7: Epoksyd
Po kilku testach z pozycją diody LED w otworach stwierdziłem, że najlepszą dyskusję osiąga się, gdy diody LED znajdują się w odległości około 0,2 cala od końca otworu. Gdy spróbujesz tego samodzielnie, jasność diod LED będzie bardzo różna w każdy otwór. Nie martw się o to; naprawimy to w kodzie. Wynika to z rodzaju użytego wiertła. Gdybym miał to zrobić ponownie, użyłbym wiertła z końcówką kulkową do otworów Ale w każdym razie, aby uzyskać odległość, zmieszałem trochę żywicy epoksydowej i włożyłem trochę w każdy otwór.
Krok 8: Składanie wszystkiego razem
Diody LED zostaną umieszczone, zaczynając od pozycji wskazówki na godzinie 12 poruszającej się w kierunku przeciwnym do ruchu wskazówek zegara przez wszystkie pozycje wskazówki godzinowej, a następnie do wskazówki minutowej, ponownie przechodząc od znaku 60 minut poruszającego się w kierunku przeciwnym do ruchu wskazówek zegara. Dzieje się tak, że gdy patrzymy od przodu, wzór diody wydaje się iść w kierunku zgodnym z ruchem wskazówek zegara.
Po utwardzeniu żywicy epoksydowej przez godzinę dodałem trochę więcej żywicy epoksydowej. Tym razem diody LED umieściłem w otworach, upewniając się, że przewody i złącza lutowane pokryłem żywicą epoksydową. Zapewnia to dobre rozproszenie światła i zabezpiecza przewody.
Krok 9: Kod
Kod znajduje się na GitHub, możesz go modyfikować do własnego użytku. Gdy włączysz wszystkie diody LED na tym samym poziomie, jasność wpadającego światła będzie bardzo różna w każdym otworze. Wynika to z różnej grubości drewna w otworach i różnicy w odcieniu drewna. Jak widać kolor drewna w moim kawałku jest dość różny. Aby zniwelować tę różnicę w jasności, wykonałem matrycę poziomów jasności led. I zmniejszono jasność jaśniejszych diod LED. Jest to proces prób i błędów, który może potrwać kilka minut, ale wyniki są tego warte.
sklejkaClock.ino
// Zegar ze sklejki |
// Autor: tinkrmind |
// Uznanie autorstwa 4.0 Międzynarodowe (CC BY 4.0). Możesz: |
// Udostępnij - kopiuj i rozpowszechniaj materiał na dowolnym nośniku lub formacie |
// Dostosuj - remiksuj, przekształcaj i buduj na materiale w dowolnym celu, nawet komercyjnym. |
// Hurra! |
#włączać |
#include"RTClib.h" |
RTC_DS3231 RTC; |
#include"Adafruit_NeoPixel.h" |
#ifdef _AVR_ |
#włączać |
#endif |
#definiuj PIN6 |
Pasek Adafruit_NeoPixel = Adafruit_NeoPixel(60, PIN, NEO_GRB + NEO_KHZ800); |
int godzinaPixel = 0; |
int minutaPiksel = 0; |
unsignedlong lastRtcCheck; |
String inputString = ""; // ciąg do przechowywania przychodzących danych |
boolean stringComplete = false; // czy napis jest kompletny |
poziom wewnętrzny[24] = {31, 51, 37, 64, 50, 224, 64, 102, 95, 255, 49, 44, 65, 230, 80, 77, 102, 87, 149, 192, 67, 109, 68, 77}; |
pustka () { |
#ifndef ESP8266 |
podczas (!Serial); // dla Leonarda/Mikro/Zero |
#endif |
// To jest dla Trinket 5V 16MHz, możesz usunąć te trzy linie, jeśli nie używasz Trinket |
#jeśli zdefiniowano (_AVR_ATtiny85_) |
if (F_CPU == 16000000) clock_prescale_set(clock_div_1); |
#endif |
// Koniec kodu specjalnego drobiazgu |
Serial.początek(9600); |
strip.początek(); |
strip.pokaż(); // Zainicjuj wszystkie piksele na „wyłączone” |
if (! rtc.begin()) { |
Serial.println("Nie można znaleźć RTC"); |
natomiast (1); |
} |
pinMode(2, INPUT_PULLUP); |
// rtc.adjust(DataCzas(F(_DATA_), F(_CZAS_))); |
if (rtc.lostPower()) { |
Serial.println("RTC stracił zasilanie, ustawmy czas!"); |
// kolejna linia ustawia RTC na datę i godzinę skompilowania tego szkicu |
rtc.adjust(DataCzas(F(_DATA_), F(_CZAS_))); |
// Ta linia ustawia RTC z wyraźną datą i godziną, na przykład, aby ustawić |
// 21 stycznia 2014 o 3 nad ranem zadzwoniłbyś: |
// rtc.adjust(DateTime(2017, 11, 06, 2, 49, 0)); |
} |
// rtc.adjust(DateTime(2017, 11, 06, 2, 49, 0)); |
//EvenEven(); |
// gdy (1); |
lastRtcCheck = 0; |
} |
pusta pętla () { |
if (millis() - lastRtcCheck >2000) { |
DataCzas teraz = rtc.now(); |
Serial.print(teraz.godzina(), DEC); |
Serial.print(':'); |
Serial.print(teraz.minuta(), DEC); |
Serial.print(':'); |
Serial.print(teraz.second(), DEC); |
Serial.println(); |
czas na przedstawienie(); |
lastRtcCheck = millis(); |
} |
jeśli (!digitalRead(2)) { |
lightUpEven(); |
} |
if (stringComplete) { |
Serial.println(ciąg wejściowy); |
if (inputString[0] == 'l') { |
Serial.println("Poziom"); |
lightUpEven(); |
} |
if (inputString[0] == 'c') { |
Serial.println("Pokazuje czas"); |
czas na przedstawienie(); |
strip.pokaż(); |
} |
if (inputString[0] == '1') { |
Serial.println("Włączanie wszystkich diod LED"); |
lightUp(strip. Color(255, 255, 255)); |
strip.pokaż(); |
} |
if (inputString[0] == '0') { |
Serial.println("Pasek kasujący"); |
jasne(); |
strip.pokaż(); |
} |
// #3, 255 ustawi diodę numer 3 na poziom 255, 255, 255 |
if (inputString[0] == '#') { |
Temp. struny; |
temp = inputString.substring(1); |
int pixNum = temp.toInt(); |
temp = inputString.substring(inputString.indexOf(', ') + 1); |
int intensywność = temp.toInt(); |
Serial.print("Ustawienie "); |
Serial.print(PixNum); |
Serial.print("do poziomu"); |
Serial.println(intensywność); |
strip.setPixelColor(pixNum, strip. Color(intensywność, intensywność, intensywność)); |
strip.pokaż(); |
} |
// #3, 255, 0, 125 ustawi diodę numer 3 na 255, 0, 125 |
if (inputString[0] == '$') { |
Temp. struny; |
temp = inputString.substring(1); |
int pixNum = temp.toInt(); |
int rIndex = inputString.indexOf(', ') + 1; |
temp = inputString.substring(rIndex); |
int rIntensity = temp.toInt(); |
intgIndex = inputString.indexOf(', ', rIndex + 1) + 1; |
temp = inputString.substring(gIndex); |
intgIntensity = temp.toInt(); |
int bIndex = inputString.indexOf(', ', gIndex + 1) + 1; |
temp = inputString.substring(bIndex); |
int bIntensity = temp.toInt(); |
Serial.print("Ustawienie "); |
Serial.print(PixNum); |
Serial.print("R do"); |
Serial.print(rIntensywność); |
Serial.print("G do"); |
Serial.print(gIntensywność); |
Serial.print(" B do "); |
Serial.println(bIntensywność); |
strip.setPixelColor(pixNum, strip. Color(rIntensity, gIntensity, bIntensity)); |
strip.pokaż(); |
} |
if (inputString[0] == 's') { |
Temp. struny; |
int godzina, minuta; |
temp = inputString.substring(1); |
godzina = temp.doInt(); |
int rIndex = inputString.indexOf(', ') + 1; |
temp = inputString.substring(rIndex); |
minuta = temp.doInt(); |
Serial.print("Pokazuje czas: "); |
Serial.print(godzina); |
Serial.print(":"); |
Serial.print(minuta); |
showTime (godzina, minuta); |
opóźnienie (1000); |
} |
inputString = ""; |
stringComplete = fałsz; |
} |
// opóźnienie(1000); |
} |
voidserialEvent() { |
while (Serial.available()) { |
char inChar = (char)Serial.read(); |
inputString += inChar; |
if (inChar == '\n') { |
stringComplete = prawda; |
} |
opóźnienie(1); |
} |
} |
voidclear() { |
for (uint16_t i = 0; i < strip.numPixels(); i++) { |
strip.setPixelColor(i, strip. Color(0, 0, 0)); |
} |
} |
voidshowTime() { |
DataCzas teraz = rtc.now(); |
hourPixel = now.hour() % 12; |
minutaPixel = (teraz.minuta() / 5) % 12 + 12; |
jasne(); |
// strip.setPixelColor(hourPixel, strip. Color(40 + 40 * level[hourPixel], 30 + 30 * level[hourPixel], 20 + 20 * level[hourPixel])); |
// strip.setPixelColor(minutePixel, strip. Color(40 + 40 * level[minutePixel], 30 + 30 * level[minutePixel], 20 + 20 * level[minutePixel])); |
strip.setPixelColor(hourPixel, strip. Color(level[hourPixel], level[hourPixel], level[hourPixel])); |
strip.setPixelColor(minutaPixel, strip. Color(poziom[minutaPixel], poziom[minutaPixel], poziom[minutaPixel])); |
// lightUp(strip. Color(255, 255, 255)); |
strip.pokaż(); |
} |
voidshowTime(int godzina,int minuta) { |
hourPixel = godzina % 12; |
minutaPiksel = (minuta / 5) % 12 + 12; |
jasne(); |
// strip.setPixelColor(hourPixel, strip. Color(40 + 40 * level[hourPixel], 30 + 30 * level[hourPixel], 20 + 20 * level[hourPixel])); |
// strip.setPixelColor(minutePixel, strip. Color(40 + 40 * level[minutePixel], 30 + 30 * level[minutePixel], 20 + 20 * level[minutePixel])); |
strip.setPixelColor(hourPixel, strip. Color(level[hourPixel], level[hourPixel], level[hourPixel])); |
strip.setPixelColor(minutaPixel, strip. Color(poziom[minutaPixel], poziom[minutaPixel], poziom[minutaPixel])); |
// lightUp(strip. Color(255, 255, 255)); |
strip.pokaż(); |
} |
voidlightUp(uint32_t kolor) { |
for (uint16_t i = 0; i < strip.numPixels(); i++) { |
strip.setPixelColor(i, kolor); |
} |
strip.pokaż(); |
} |
voidlightUpEven() { |
for (uint16_t i = 0; i < strip.numPixels(); i++) { |
strip.setPixelColor(i, strip. Color(poziom, poziom, poziom)); |
} |
strip.pokaż(); |
} |
zobacz rawplywoodClock.ino hostowane z ❤ przez GitHub
Krok 10: Widzenie komputerowe - kalibracja
Świadomie zdecydowałem się nie używać w tym projekcie forniru. Gdybym miał, grubość drewna byłaby taka sama przed wszystkimi diodami LED. Ale ponieważ mam różną grubość drewna przed każdą diodą LED i ponieważ kolor drewna również bardzo się różni, jasność diody LED jest inna dla każdej diody LED. Aby wszystkie diody LED wydawały się mieć tę samą jasność, wymyśliłem sprytną sztuczkę.
Napisałem kod przetwarzania (na GitHubie), który robi zdjęcie zegara i analizuje kolejno jasność każdej diody LED. Następnie zmienia moc każdej diody LED, aby spróbować uzyskać tę samą jasność, co najciemniejsza dioda LED. Teraz wiem, że to przesada, ale przetwarzanie obrazu to świetna zabawa! I mam nadzieję, że opracuję kod kalibracyjny jako bibliotekę.
Na powyższych zdjęciach widać jasność diod LED przed i po kalibracji.
kalibracjaDispllay.pde
importprzetwarzanie.wideo.*; |
importprocessing.serial.*; |
port szeregowy myPort; |
Przechwytywanie wideo; |
ostateczna liczba =24; |
int ledNum =0; |
// musisz mieć te globalne zmienne, aby użyć PxPGetPixelDark() |
int rCiemny, gCiemny, bCiemny, aCiemny; |
int rLed, gLed, bled, aLed; |
int rOrg, gOrg, bOrg, aOrg; |
int rTemp, gTemp, bTemp, aTemp; |
PObraz naszObraz; |
int runNumber =0; |
int akceptowalnyBłąd =3; |
int zrobione; |
int numPixelsInLed; |
long ledIntensywność; |
int ledPower; |
długi targetIntensity =99999999; |
voidsetup() { |
gotowe = newint[liczbaLed]; |
numPixelsInLed =newint[numLed]; |
ledIntensity =newlong[numLed]; |
ledPower =newint[liczbaLed]; |
for (int i=0; i<liczba; i++) { |
moc diody =255; |
} |
printArray(Serial.list()); |
Ciąg nazwa_portu =Serial.list()[31]; |
mójPort =nowySeryjny(ten, nazwa_portu, 9600); |
rozmiar (640, 480); |
video = newCapture(to, szerokość, wysokość); |
wideo.start(); |
noStroke(); |
gładki(); |
opóźnienie (1000); // Poczekaj na otwarcie portu szeregowego |
} |
nieważne rysowanie () { |
jeśli (wideo.dostępne()) { |
if (done[ledNum] ==0) { |
wyczyśćWyświetlacz(); |
opóźnienie (1000); |
wideo.odczyt(); |
obraz(wideo, 0, 0, szerokość, wysokość); // Narysuj wideo z kamery internetowej na ekran |
saveFrame("dane/no_leds.jpg"); |
if (runNumber !=0) { |
if ((ledIntensity[ledNum] - targetIntensity)*100/targetIntensity > acceptError) { |
ledPower[ledNum] -=pow(0.75, runNum)*100+1; |
} |
if ((targetIntensity - ledIntensity[ledNum])*100/targetIntensity > acceptError) { |
ledPower[ledNum] +=pow(0.75, runNum)*100+1; |
} |
if (abs(targetIntensity - ledIntensity[ledNum])*100/targetIntensity <= acceptError) { |
gotowe[liczba_zad] =1; |
print("Dioda "); |
print(ledNum); |
print("gotowe"); |
} |
if (ledPower[ledNum] >255) { |
mocled[liczbaled] =255; |
} |
if (moc diody[liczba diod] <0) { |
mocled[liczbaled]=0; |
} |
} |
setLedPower(ledNum, ledPower[ledNum]); |
opóźnienie (1000); |
wideo.odczyt(); |
obraz(wideo, 0, 0, szerokość, wysokość); // Narysuj wideo z kamery internetowej na ekran |
opóźnienie(10); |
while (mójPort.dostępny() >0) { |
int inByte = myPort.read(); |
//print(char(inByte)); |
} |
Ciąg nazwa obrazu ="dane/"; |
imageName+=str(ledNum); |
imageName +="_led.jpg"; |
saveFrame(nazwaobrazu); |
String originalImageName ="data/org"; |
originalImageName+=str(ledNum); |
oryginalna nazwa obrazu +=".jpg"; |
if (runNumber ==0) { |
saveFrame(oryginalnaNazwaObrazu); |
} |
PImage noLedImg =loadImage("dane/brak_led.jpg"); |
PImage ledImg =loadImage(imageName); |
PImage originalImg =loadImage(oryginalnaNazwaObrazu); |
noLedImg.loadPixels(); |
ledImg.loadPixels(); |
oryginalnyImg.loadPixels(); |
tło (0); |
wczytaj piksele(); |
ledIntensity[ledNum] =0; |
numPixelsInLed[ledNum] =0; |
for (int x =0; x<szerokość; x++) { |
for (int y =0; y<wysokość; y++) { |
PxPGetPixelDark(x, y, noLedImg.piksele, szerokość); |
PxPGetPixelLed(x, y, ledImg.piksele, szerokość); |
PxPGetPixelOrg(x, y, oryginalnyImg.piksele, szerokość); |
jeśli ((rOrg+gOrg/2+bOrg/3)-(rDark+gDark/2+bDark/3) >75) { |
ledIntensity[ledNum] = ledIntensity[ledNum] +(rLed+gLed/2+bLed/3) -(rCiemny+gCiemny/2+bCiemny/3); |
rTemp=255; |
gTemp=255; |
bTemp=255; |
numPixelsInLed[ledNum]++; |
} w przeciwnym razie { |
rTemp=0; |
gTemp=0; |
bTemp=0; |
} |
PxPSetPixel(x, y, rTemp, gTemp, bTemp, 255, piksele, szerokość); |
} |
} |
ledIntensity[ledNum] /= numPixelsInLed[ledNum]; |
if (targetIntensity > ledIntensity[ledNum] && runNumber ==0) { |
targetIntensity = ledIntensity[ledNum]; |
} |
zaktualizuj piksele(); |
} |
print(ledNum); |
print(', '); |
print(ledPower[ledNum]); |
print(', '); |
println(ledIntensity[ledNum]); |
ledNum++; |
if (ledNum == numLed) { |
int donezo =0; |
for (int i=0; i<liczba; i++) { |
donezo += gotowe; |
} |
if (donezo == liczba) { |
println("GOTOWE"); |
for (int i=0; i<liczba; i++) { |
drukuj(i); |
print("\t"); |
println(ledPower); |
} |
print("int poziom["); |
print(ledNum); |
print("] = {"); |
for (int i=0; i<numLed-1; i++) { |
print(ledPower); |
print(', '); |
} |
print(ledPower[numLed -1]); |
println("};"); |
lightUpEven(); |
podczas (prawda); |
} |
print("Intensywność docelowa: "); |
if (runNumber ==0) { |
celIntensywność -=1; |
} |
println(intensywność docelowa); |
liczba led =0; |
runNumer++; |
} |
} |
} |
voidPxPGetPixelOrg(intx, inty, int pixelArray, intpixelsWidth) { |
int thisPixel=pixelArray[x+y*pixelsWidth]; // pobieranie kolorów jako int z pikseli |
aOrg = (ten piksel >>24) &0xFF; // musimy przesunąć i zamaskować, aby uzyskać każdy komponent osobno |
rOrg = (ten piksel >>16) &0xFF; // to jest szybsze niż wywołanie red(), green(), blue() |
gOrg = (ten piksel >>8) &0xFF; |
bOrg = tenPixel &0xFF; |
} |
voidPxPGetPixelDark(intx, inty, int pixelArray, intpixelsWidth) { |
int thisPixel=pixelArray[x+y*pixelsWidth]; // pobieranie kolorów jako int z pikseli |
aCiemny = (ten piksel >>24) &0xFF; // musimy przesunąć i zamaskować, aby uzyskać każdy komponent osobno |
rCiemny = (ten piksel >>16) &0xFF; // to jest szybsze niż wywołanie red(), green(), blue() |
gDark = (ten piksel >>8) &0xFF; |
bDark = tenPixel &0xFF; |
} |
voidPxPGetPixelLed(intx, inty, int pixelArray, intpixelsWidth) { |
int thisPixel=pixelArray[x+y*pixelsWidth]; // pobieranie kolorów jako int z pikseli |
aLed = (tenPixel >>24) &0xFF; // musimy przesunąć i zamaskować, aby uzyskać każdy komponent osobno |
rLed = (ten piksel >>16) &0xFF; // to jest szybsze niż wywołanie red(), green(), blue() |
gLed = (ten piksel >>8) &0xFF; |
bLed = tenPixel &0xFF; |
} |
voidPxPSetPixel(intx, inty, intr, intg, intb, inta, int pixelArray, intpixelsWidth) { |
a =(a <<24); |
r = r <<16; // Pakujemy wszystkie 4 komponenty w jeden int |
g = g <<8; // więc musimy je przenieść na swoje miejsca |
kolor argb = a | r | g | b; // binarna operacja "lub" dodaje je wszystkie do jednego int |
pixelArray[x+y*pixelsWidth]= argb; // w końcu ustawiamy int z kolorami te w pikselach |
} |
zobacz rawscaleDispllay.pde hostowany z ❤ przez GitHub
Krok 11: Pożegnalne uwagi
Pułapki, których należy unikać:
* Z drewnem dostajesz to, za co płacisz. Więc zdobądź drewno dobrej jakości. Sklejka brzozowa to dobry wybór; każde lekkie lite drewno również się sprawdzi. Opuściłem drewno i żałuję swojej decyzji.
* Lepiej wiercić mniej niż więcej. Kilka dziur poszło zbyt głęboko dla mojego kawałka. A żywica epoksydowa prześwituje na przedniej powierzchni. Jest to bardzo zauważalne, gdy to zauważysz.
* Użyj wiertła z końcówką kulistą zamiast prostego końca. Nie eksperymentowałem z końcówką kulkową, ale jestem pewien, że wyniki będą znacznie lepsze.
Flirtuję z pomysłem sprzedawania ich na Etsy lub Tindie. Byłbym bardzo wdzięczny, gdybyś mógł skomentować poniżej, jeśli uważasz, że ma to sens:)