summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--keyboards/ut47/LED_controls.ino420
-rw-r--r--keyboards/ut47/config.h87
-rw-r--r--keyboards/ut47/keymaps/default/config.h24
-rw-r--r--keyboards/ut47/keymaps/default/keymap.c88
-rw-r--r--keyboards/ut47/keymaps/default/readme.md19
-rw-r--r--keyboards/ut47/led.c38
-rw-r--r--keyboards/ut47/matrix.c209
-rw-r--r--keyboards/ut47/readme.md41
-rw-r--r--keyboards/ut47/rules.mk70
-rw-r--r--keyboards/ut47/ut47.c50
-rw-r--r--keyboards/ut47/ut47.h49
11 files changed, 1095 insertions, 0 deletions
diff --git a/keyboards/ut47/LED_controls.ino b/keyboards/ut47/LED_controls.ino
new file mode 100644
index 0000000000..dd50300eb7
--- /dev/null
+++ b/keyboards/ut47/LED_controls.ino
@@ -0,0 +1,420 @@
+#include <FastGPIO.h>
+#include <TimerOne.h>
+
+int iByte;
+byte col = 0;
+byte leds[12][4];
+byte pass = 1;
+int fadecount = 1;
+const int fadelimit = 3000;
+const int fadelimitshort = 1000;
+byte mode = 4;
+byte brightness = 2;
+boolean changemode = 0;
+int rain = 0;
+const int rainlimit = 5000;
+const int rainfade = 5000;
+byte rx = 0;
+byte ry = 0;
+
+// pin[xx] on led matrix connected to nn on Arduino (-1 is dummy to make array start at pos 1)
+int pins[17] = {
+ -1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 16, 14, 15, 18, 19, 20, 21
+};
+
+// col[xx] of leds = pin yy on led matrix
+int cols[12] = {
+ pins[8], pins[7], pins[6], pins[5], pins[9], pins[10], pins[11], pins[12], pins[13], pins[14], pins[15], pins[16]
+};
+
+// row[xx] of leds = pin yy on led matrix
+int rows[4] = {
+ pins[1], pins[2], pins[3], pins[4]
+};
+
+
+#define DELAY 0
+extern byte leds[12][4];
+
+void setup() {
+ Serial1.begin(9600);
+ setupLeds();
+ for (int s = 0; s < 5; s++) {
+ for ( int r = 1; r < 9; r++) {
+ delayMicroseconds(65000);
+ delayMicroseconds(65000);
+ for (int j = 0; j < 4; j++) {
+ for (int i = 0; i < 12; i++) {
+ leds[i][j] = 1;
+ for (int p = 0; p < 25; p++) {
+ }
+ leds[i][j] = r;
+ }
+ }
+ }
+ for ( int r = 9; r > 0; r--) {
+ delayMicroseconds(65000);
+ delayMicroseconds(65000);
+ delayMicroseconds(65000);
+ for (int j = 0; j < 4; j++) {
+ for (int i = 0; i < 12; i++) {
+ leds[i][j] = 1;
+ for (int p = 0; p < 25; p++) {
+ }
+ leds[i][j] = r;
+ }
+ }
+ }
+ }
+}
+
+void loop() {
+
+ switch (mode) {
+ case 0:
+ //Blacklight
+ for (int i = 0; i < 12; i++) {
+ for (int j = 0; j < 4; j++) {
+ leds[i][j] = brightness;
+ }
+ }
+ checkserial();
+ break;
+ case 1:
+ //Breathing
+ for ( int r = 1; r < 9; r++) {
+ checkserial();
+ if (changemode == 0) {
+ delayMicroseconds(65000);
+ delayMicroseconds(65000);
+ delayMicroseconds(65000);
+ for (int j = 0; j < 4; j++) {
+ for (int i = 0; i < 12; i++) {
+ leds[i][j] = 1;
+ for (int p = 0; p < 25; p++) {
+ }
+ leds[i][j] = r;
+ }
+ }
+ }
+ else {
+ break;
+ }
+ }
+ for ( int r = 9; r > 0; r--) {
+ checkserial();
+ if (changemode == 0) {
+ delayMicroseconds(65000);
+ delayMicroseconds(65000);
+ delayMicroseconds(65000);
+ delayMicroseconds(65000);
+ for (int j = 0; j < 4; j++) {
+ for (int i = 0; i < 12; i++) {
+ leds[i][j] = 1;
+ for (int p = 0; p < 25; p++) {
+ }
+ leds[i][j] = r;
+ }
+ }
+ }
+ else {
+ break;
+ }
+ }
+ for ( int r = 1; r < 30; r++) {
+ checkserial();
+ if (changemode == 0) {
+ delayMicroseconds(65000);
+ delayMicroseconds(65000);
+ }
+ else {
+ break;
+ }
+ }
+ break;
+ case 2:
+ //Random
+ leds[random(12)][random(4)] = random(8);
+ delayMicroseconds(10000);
+ checkserial();
+ break;
+ case 3:
+ //Rain
+ rain++;
+ if (rain > rainlimit) {
+ rain = 0;
+ rx = random(12);
+ ry = random(4);
+ if (leds[rx][ry] == 0) {
+ leds[rx][ry] = 18;
+ }
+ }
+ fadecount++;
+ if (fadecount > rainfade) {
+ fadecount = 1;
+ for (int i = 0; i < 12; i++) {
+ for (int j = 0; j < 4; j++) {
+ if (leds[i][j] > 0) {
+ leds[i][j] = leds[i][j] - 1;
+ }
+ }
+ }
+ }
+ checkserial();
+ break;
+ case 4:
+ //Reactive
+ fadecount++;
+ if (fadecount > fadelimit) {
+ fadecount = 1;
+ for (int i = 0; i < 12; i++) {
+ for (int j = 0; j < 4; j++) {
+ if (leds[i][j] > 0) {
+ leds[i][j] = leds[i][j] - 1;
+ }
+ }
+ }
+ }
+ checkserial();
+ break;
+ case 5:
+ //Reactive Target
+ fadecount++;
+ if (fadecount > fadelimitshort) {
+ fadecount = 1;
+ for (int i = 0; i < 12; i++) {
+ for (int j = 0; j < 4; j++) {
+ if (leds[i][j] > 0) {
+ leds[i][j] = leds[i][j] - 1;
+ }
+ }
+ }
+ }
+ checkserial();
+ break;
+ default:
+ mode = 0;
+ break;
+ }
+ changemode = 0;
+}
+
+void checkserial() {
+ if (Serial1.available() > 0) {
+ iByte = Serial1.read();
+ if (iByte == 100) {
+ brightness++;
+ if (brightness > 9) {
+ brightness = 1;
+ }
+ }
+ if (iByte == 101) {
+ mode++;
+ }
+ if (iByte < 100) {
+ if (mode == 4) {
+ byte row = iByte / 16;
+ byte col = iByte % 16;
+ leds[col][row] = 18;
+ }
+ if (mode == 5) {
+ byte row = iByte / 16;
+ byte col = iByte % 16;
+ for (byte i = 0; i < 12; i++) {
+ leds[i][row] = 18;
+ }
+ for (byte p = 0; p < 4; p++) {
+ leds[col][p] = 18;
+ }
+ }
+ }
+ }
+}
+
+void setupLeds() {
+ // sets the pins as output
+ FastGPIO::Pin<2>::setOutputLow();
+ FastGPIO::Pin<3>::setOutputLow();
+ FastGPIO::Pin<4>::setOutputLow();
+ FastGPIO::Pin<5>::setOutputLow();
+ FastGPIO::Pin<6>::setOutputLow();
+ FastGPIO::Pin<7>::setOutputLow();
+ FastGPIO::Pin<8>::setOutputLow();
+ FastGPIO::Pin<9>::setOutputLow();
+ FastGPIO::Pin<10>::setOutputLow();
+ FastGPIO::Pin<16>::setOutputLow();
+ FastGPIO::Pin<14>::setOutputLow();
+ FastGPIO::Pin<15>::setOutputLow();
+ FastGPIO::Pin<18>::setOutputLow();
+ FastGPIO::Pin<19>::setOutputLow();
+ FastGPIO::Pin<20>::setOutputLow();
+ FastGPIO::Pin<21>::setOutputLow();
+
+ // set up Cols
+ FastGPIO::Pin<6>::setOutputValueLow();
+ FastGPIO::Pin<7>::setOutputValueLow();
+ FastGPIO::Pin<8>::setOutputValueLow();
+ FastGPIO::Pin<9>::setOutputValueLow();
+ FastGPIO::Pin<10>::setOutputValueLow();
+ FastGPIO::Pin<16>::setOutputValueLow();
+ FastGPIO::Pin<14>::setOutputValueLow();
+ FastGPIO::Pin<15>::setOutputValueLow();
+ FastGPIO::Pin<18>::setOutputValueLow();
+ FastGPIO::Pin<19>::setOutputValueLow();
+ FastGPIO::Pin<20>::setOutputValueLow();
+ FastGPIO::Pin<21>::setOutputValueLow();
+
+ // set up Rows
+ FastGPIO::Pin<2>::setOutputValueLow();
+ FastGPIO::Pin<3>::setOutputValueLow();
+ FastGPIO::Pin<4>::setOutputValueLow();
+ FastGPIO::Pin<5>::setOutputValueLow();
+
+ clearLeds();
+ Timer1.initialize(25);
+ Timer1.attachInterrupt(display);
+
+}
+
+void clearLeds() {
+ // Clear display array
+ for (int i = 0; i < 12; i++) {
+ for (int j = 0; j < 4; j++) {
+ leds[i][j] = 0;
+ }
+ }
+}
+
+void onLeds() {
+ // Clear display array
+ for (int i = 0; i < 12; i++) {
+ for (int j = 0; j < 4; j++) {
+ leds[i][j] = 7;
+ }
+ }
+}
+
+// Interrupt routine
+void display() {
+
+ switch (col) { // Turn whole previous column off
+ case 0:
+ FastGPIO::Pin<6>::setOutputValueLow();
+ break;
+ case 1:
+ FastGPIO::Pin<7>::setOutputValueLow();
+ break;
+ case 2:
+ FastGPIO::Pin<8>::setOutputValueLow();
+ break;
+ case 3:
+ FastGPIO::Pin<9>::setOutputValueLow();
+ break;
+ case 4:
+ FastGPIO::Pin<10>::setOutputValueLow();
+ break;
+ case 5:
+ FastGPIO::Pin<16>::setOutputValueLow();
+ break;
+ case 6:
+ FastGPIO::Pin<14>::setOutputValueLow();
+ break;
+ case 7:
+ FastGPIO::Pin<15>::setOutputValueLow();
+ break;
+ case 8:
+ FastGPIO::Pin<18>::setOutputValueLow();
+ break;
+ case 9:
+ FastGPIO::Pin<19>::setOutputValueLow();
+ break;
+ case 10:
+ FastGPIO::Pin<20>::setOutputValueLow();
+ break;
+ case 11:
+ FastGPIO::Pin<21>::setOutputValueLow();
+ break;
+ }
+
+ col++;
+ if (col == 12) {
+ col = 0;
+ pass++;
+ if (pass > 8) {
+ pass = 1;
+ }
+ }
+ for (int row = 0; row < 4; row++) {
+ if (leds[col][row] > pass) {
+ switch (row) { // Turn on this led
+ case 0:
+ FastGPIO::Pin<2>::setOutputValueLow();
+ break;
+ case 1:
+ FastGPIO::Pin<3>::setOutputValueLow();
+ break;
+ case 2:
+ FastGPIO::Pin<4>::setOutputValueLow();
+ break;
+ case 3:
+ FastGPIO::Pin<5>::setOutputValueLow();
+ break;
+ }
+ }
+ else {
+ switch (row) { // Turn off this led
+ case 0:
+ FastGPIO::Pin<2>::setOutputValueHigh();
+ break;
+ case 1:
+ FastGPIO::Pin<3>::setOutputValueHigh();
+ break;
+ case 2:
+ FastGPIO::Pin<4>::setOutputValueHigh();
+ break;
+ case 3:
+ FastGPIO::Pin<5>::setOutputValueHigh();
+ break;
+ }
+ }
+ }
+ switch (col) { // Turn column on
+ case 0:
+ FastGPIO::Pin<6>::setOutputValueHigh();
+ break;
+ case 1:
+ FastGPIO::Pin<7>::setOutputValueHigh();
+ break;
+ case 2:
+ FastGPIO::Pin<8>::setOutputValueHigh();
+ break;
+ case 3:
+ FastGPIO::Pin<9>::setOutputValueHigh();
+ break;
+ case 4:
+ FastGPIO::Pin<10>::setOutputValueHigh();
+ break;
+ case 5:
+ FastGPIO::Pin<16>::setOutputValueHigh();
+ break;
+ case 6:
+ FastGPIO::Pin<14>::setOutputValueHigh();
+ break;
+ case 7:
+ FastGPIO::Pin<15>::setOutputValueHigh();
+ break;
+ case 8:
+ FastGPIO::Pin<18>::setOutputValueHigh();
+ break;
+ case 9:
+ FastGPIO::Pin<19>::setOutputValueHigh();
+ break;
+ case 10:
+ FastGPIO::Pin<20>::setOutputValueHigh();
+ break;
+ case 11:
+ FastGPIO::Pin<21>::setOutputValueHigh();
+ break;
+ }
+
+}
diff --git a/keyboards/ut47/config.h b/keyboards/ut47/config.h
new file mode 100644
index 0000000000..4335e3edd5
--- /dev/null
+++ b/keyboards/ut47/config.h
@@ -0,0 +1,87 @@
+/*
+Copyright 2018 Carlos Filoteo
+
+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 2 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.
+
+You should have received a copy of the GNU General Public License
+along with this program. If not, see <http://www.gnu.org/licenses/>.
+*/
+
+#ifndef CONFIG_H
+#define CONFIG_H
+
+#include "config_common.h"
+
+/* USB Device descriptor parameter */
+#define VENDOR_ID 0xFEED
+#define PRODUCT_ID 0x0000
+#define DEVICE_VER 0x0001
+#define MANUFACTURER 40percent.club
+#define PRODUCT ut47
+#define DESCRIPTION An awesome 40% keyboard
+
+/* key matrix size */
+#define MATRIX_ROWS 4
+#define MATRIX_COLS 12
+
+#define MATRIX_ROW_PINS { D1, D0, D4, C6 }
+#define MATRIX_COL_PINS { D7, E6, B4, B5, B6, B2, B3, B1, F7, F6, F5, F4 }
+#define UNUSED_PINS
+
+/* COL2ROW, ROW2COL, or CUSTOM_MATRIX */
+#define DIODE_DIRECTION COL2ROW
+
+// #define BACKLIGHT_PIN B7
+// #define BACKLIGHT_BREATHING
+// #define BACKLIGHT_LEVELS 3
+
+/* Debounce reduces chatter (unintended double-presses) - set 0 if debouncing is not needed */
+#define DEBOUNCING_DELAY 5
+
+/* define if matrix has ghost (lacks anti-ghosting diodes) */
+//#define MATRIX_HAS_GHOST
+
+/* Mechanical locking support. Use KC_LCAP, KC_LNUM or KC_LSCR instead in keymap */
+#define LOCKING_SUPPORT_ENABLE
+/* Locking resynchronize hack */
+#define LOCKING_RESYNC_ENABLE
+
+/* key combination for magic key command */
+#define IS_COMMAND() ( \
+ keyboard_report->mods == (MOD_BIT(KC_LSHIFT) | MOD_BIT(KC_RSHIFT)) \
+)
+
+/* Enable GNAP matrix serial output */
+#define GNAP_ENABLE
+
+/* USART configuration */
+#ifdef __AVR_ATmega32U4__
+# define SERIAL_UART_BAUD 9600
+# define SERIAL_UART_DATA UDR1
+# define SERIAL_UART_UBRR (F_CPU / (16UL * SERIAL_UART_BAUD) - 1)
+# define SERIAL_UART_RXD_VECT USART1_RX_vect
+# define SERIAL_UART_TXD_READY (UCSR1A & _BV(UDRE1))
+# define SERIAL_UART_INIT() do { \
+ /* baud rate */ \
+ UBRR1L = SERIAL_UART_UBRR; \
+ /* baud rate */ \
+ UBRR1H = SERIAL_UART_UBRR >> 8; \
+ /* enable TX */ \
+ UCSR1B = _BV(TXEN1); \
+ /* 8-bit data */ \
+ UCSR1C = _BV(UCSZ11) | _BV(UCSZ10); \
+ sei(); \
+ } while(0)
+# else
+# error "USART configuration is needed."
+#endif
+
+#endif
diff --git a/keyboards/ut47/keymaps/default/config.h b/keyboards/ut47/keymaps/default/config.h
new file mode 100644
index 0000000000..46098a22fb
--- /dev/null
+++ b/keyboards/ut47/keymaps/default/config.h
@@ -0,0 +1,24 @@
+/* Copyright 2018 Carlos Filoteo
+ *
+ * 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 2 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.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ */
+
+#ifndef CONFIG_USER_H
+#define CONFIG_USER_H
+
+#include "config_common.h"
+
+// place overrides here
+
+#endif
diff --git a/keyboards/ut47/keymaps/default/keymap.c b/keyboards/ut47/keymaps/default/keymap.c
new file mode 100644
index 0000000000..a02543dacb
--- /dev/null
+++ b/keyboards/ut47/keymaps/default/keymap.c
@@ -0,0 +1,88 @@
+/* Copyright 2018 Carlos Filoteo
+ *
+ * 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 2 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.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ */
+#include QMK_KEYBOARD_H
+#ifdef LED_ENABLE
+ #include "protocol/serial.h"
+#endif
+
+#define _______ KC_TRNS
+#define LT3_TAB LT(3, KC_TAB)
+#define MT_RSFT_ENT MT(MOD_RSFT, KC_ENT)
+
+enum custom_keycodes {
+ LED_TOG = SAFE_RANGE,
+ LED_CHG
+};
+
+const uint16_t PROGMEM keymaps[][MATRIX_ROWS][MATRIX_COLS] = {
+
+LAYOUT(
+ KC_ESC, KC_Q, KC_W, KC_E, KC_R, KC_T, KC_Y, KC_U, KC_I, KC_O, KC_P, KC_BSPC,
+ LT3_TAB, KC_A, KC_S, KC_D, KC_F, KC_G, KC_H, KC_J, KC_K, KC_L, KC_SCLN, KC_QUOT,
+ KC_LSFT, KC_Z, KC_X, KC_C, KC_V, KC_B, KC_N, KC_M, KC_COMM, KC_DOT, KC_SLSH, MT_RSFT_ENT,
+ KC_LCTL, KC_LALT, KC_LGUI, KC_APP, MO(2), KC_SPC, MO(1), KC_LEFT, KC_DOWN, KC_UP, KC_RGHT
+),
+
+LAYOUT(
+ KC_GRV, KC_1, KC_2, KC_3, KC_4, KC_5, KC_6, KC_7, KC_8, KC_9, KC_0, KC_DELETE,
+ _______, _______, _______, _______, _______, _______, _______, KC_MINS, KC_EQL, KC_LBRC, KC_RBRC, KC_BSLS,
+ _______, KC_F11, KC_F12, KC_F13, KC_F14, KC_F15, KC_F16, KC_F17, KC_F18, KC_F19, KC_F20, _______,
+ _______, _______, _______, KC_CAPS, _______, _______, _______, KC_HOME, KC_PGDN, KC_PGUP, KC_END
+),
+
+LAYOUT(
+ KC_FN6, KC_FN7, KC_FN8, KC_FN9, KC_FN10, KC_FN11, KC_FN12, KC_FN13, KC_FN14, KC_FN15, KC_FN16, KC_DELETE,
+ _______, _______, _______, _______, _______, _______, _______, KC_FN17, KC_FN18, KC_FN19, KC_FN20, KC_FN21,
+ _______, KC_F1, KC_F2, KC_F3, KC_F4, KC_F5, KC_F6, KC_F7, KC_F8, KC_F9, KC_F10, _______,
+ _______, _______, _______, KC_CAPS, _______, _______, _______, KC_HOME, KC_PGDN, KC_PGUP, KC_END
+),
+
+LAYOUT( /* Tab */
+ KC_ESC, KC_CALC, KC_WHOM, KC_MAIL, KC_MYCM, _______, _______, _______, _______, _______, KC_PSCR, _______,
+ _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______,
+ _______, LED_TOG, LED_CHG, _______, _______, _______, _______, _______, _______, _______, _______, _______,
+ _______, _______, _______, _______, _______, _______, _______, KC_MS_L, KC_MS_D, KC_MS_U, KC_MS_R
+),
+};
+
+//LED keymap functions
+ #ifdef LED_ENABLE
+void led_chmode(void) {
+ serial_send(101);
+}
+
+void led_toggle(void) {
+ serial_send(100);
+}
+
+bool process_record_user(uint16_t keycode, keyrecord_t *record) {
+ if (record->event.pressed) {
+ switch(keycode) {
+ case LED_TOG:
+ #ifdef LED_ENABLE
+ led_toggle();
+ #endif
+ return false;
+ case LED_CHG:
+ #ifdef LED_ENABLE
+ led_chmode();
+ #endif
+ return false;
+ }
+ }
+ return true;
+};
+#endif
diff --git a/keyboards/ut47/keymaps/default/readme.md b/keyboards/ut47/keymaps/default/readme.md
new file mode 100644
index 0000000000..e09dc73fab
--- /dev/null
+++ b/keyboards/ut47/keymaps/default/readme.md
@@ -0,0 +1,19 @@
+# UT47 default keymap
+
+![UT47 layout image](https://i.imgur.com/Tsz5qsF.png)
+
+[KLE](http://www.keyboard-layout-editor.com/##@@_y:0%3B&=Esc&=Q&=W&=E&=R&=T&=Y&=U&=I&=O&=P&_w:1.5%3B&=Back%20Space&_x:0.25&a:4&f:4&w:4&h:4&d:true%3B&=%3Cb%3EGNAP!%3C%2F%2Fb%3E%3Cp%3E%3Cp%3EMinimum%20stagger%3Cp%3E47%20key%20layout%3B&@_a:7&f:3&w:1.25%3B&=Tab&=A&=S&=D&=F&=G&=H&=J&=K&=L&=%2F%3B&_w:1.25%3B&=%27%3B&@_w:1.5%3B&=Shift&=Z&=X&=C&=V&=B&=N&=M&=,&=.&=%2F%2F&=Return%3B&@=Ctrl&=Alt&=Super&=Menu&_w:1.25%3B&=%2F&dArr%2F%3B&_w:2%3B&=&_w:1.25%3B&=%2F&uArr%2F%3B&=%2F&larr%2F%3B&=%2F&darr%2F%3B&=%2F&uarr%2F%3B&=%2F&rarr%2F%3B%3B&=undefined)
+
+### LED Controls
+
+Use <kbd>TAB</kbd>+<kbd>Z</kbd> to cycle through brightness (8 steps)
+
+Use <kbd>TAB</kbd>+<kbd>X</kbd> to cycle through the following LED modes:
+
+- solid
+- breathing
+- random
+- rain
+- reactive
+- poptang
+- off
diff --git a/keyboards/ut47/led.c b/keyboards/ut47/led.c
new file mode 100644
index 0000000000..9458c038fe
--- /dev/null
+++ b/keyboards/ut47/led.c
@@ -0,0 +1,38 @@
+/*
+Copyright 2012 Jun Wako <wakojun@gmail.com>
+
+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 2 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.
+
+You should have received a copy of the GNU General Public License
+along with this program. If not, see <http://www.gnu.org/licenses/>.
+*/
+
+#include <avr/io.h>
+#include "stdint.h"
+#include "led.h"
+
+
+void led_set(uint8_t usb_led)
+{
+ if (usb_led & (1<<USB_LED_CAPS_LOCK)) {
+ // output low
+ DDRB |= (1<<0);
+ PORTB &= ~(1<<0);
+ DDRD |= (1<<5);
+ PORTD &= ~(1<<5);
+ } else {
+ // Hi-Z
+ DDRB &= ~(1<<0);
+ PORTB &= ~(1<<0);
+ DDRD &= ~(1<<5);
+ PORTD &= ~(1<<5);
+ }
+}
diff --git a/keyboards/ut47/matrix.c b/keyboards/ut47/matrix.c
new file mode 100644
index 0000000000..18d420dbbb
--- /dev/null
+++ b/keyboards/ut47/matrix.c
@@ -0,0 +1,209 @@
+/*
+Copyright 2018 Carlos Filoteo
+
+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 2 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.
+
+You should have received a copy of the GNU General Public License
+along with this program. If not, see <http://www.gnu.org/licenses/>.
+*/
+
+/*
+ * scan matrix
+ */
+#include <stdint.h>
+#include <stdbool.h>
+#include <avr/io.h>
+#include <util/delay.h>
+#include "print.h"
+#include "debug.h"
+#include "util.h"
+#include "matrix.h"
+#include "protocol/serial.h"
+
+
+#ifndef DEBOUNCE
+# define DEBOUNCE 5
+#endif
+static uint8_t debouncing = DEBOUNCE;
+
+/* matrix state(1:on, 0:off) */
+static matrix_row_t matrix[MATRIX_ROWS];
+static matrix_row_t matrix_debouncing[MATRIX_ROWS];
+
+static matrix_row_t read_cols(void);
+static void init_cols(void);
+static void unselect_rows(void);
+static void select_row(uint8_t row);
+
+
+inline
+uint8_t matrix_rows(void)
+{
+ return MATRIX_ROWS;
+}
+
+inline
+uint8_t matrix_cols(void)
+{
+ return MATRIX_COLS;
+}
+
+void matrix_init(void)
+{
+ // initialize row and col
+ unselect_rows();
+ init_cols();
+
+ // initialize matrix state: all keys off
+ for (uint8_t i=0; i < MATRIX_ROWS; i++) {
+ matrix[i] = 0;
+ matrix_debouncing[i] = 0;
+ }
+
+ serial_init();
+}
+
+uint8_t matrix_scan(void)
+{
+ for (uint8_t i = 0; i < MATRIX_ROWS; i++) {
+ select_row(i);
+ _delay_us(30); // without this wait read unstable value.
+ matrix_row_t cols = read_cols();
+ if (matrix_debouncing[i] != cols) {
+ matrix_debouncing[i] = cols;
+ if (debouncing) {
+ debug("bounce!: "); debug_hex(debouncing); debug("\n");
+ }
+ debouncing = DEBOUNCE;
+ }
+ unselect_rows();
+ }
+
+ if (debouncing) {
+ if (--debouncing) {
+ _delay_ms(1);
+ } else {
+ for (uint8_t i = 0; i < MATRIX_ROWS; i++) {
+ matrix[i] = matrix_debouncing[i];
+ }
+ }
+ }
+
+ return 1;
+}
+
+bool matrix_is_modified(void)
+{
+ if (debouncing) return false;
+ return true;
+}
+
+inline
+bool matrix_is_on(uint8_t row, uint8_t col)
+{
+ return (matrix[row] & ((matrix_row_t)1<<col));
+}
+
+inline
+matrix_row_t matrix_get_row(uint8_t row)
+{
+ return matrix[row];
+}
+
+void matrix_print(void)
+{
+ print("\nr/c 0123456789ABCDEF\n");
+ for (uint8_t row = 0; row < MATRIX_ROWS; row++) {
+ phex(row); print(": ");
+ pbin_reverse16(matrix_get_row(row));
+ print("\n");
+ }
+}
+
+uint8_t matrix_key_count(void)
+{
+ uint8_t count = 0;
+ for (uint8_t i = 0; i < MATRIX_ROWS; i++) {
+ count += bitpop16(matrix[i]);
+ }
+ return count;
+}
+
+/* Column pin configuration
+ * col: 0 1 2 3 4 5 6 7 8 9 10 11
+ * pin: D7 E6 B4 B5 B6 B2 B3 B1 F7 F6 F5 F4
+ */
+
+static void init_cols(void)
+{
+ // Input with pull-up(DDR:0, PORT:1)
+ DDRF &= ~(1<<4 | 1<<5 | 1<<6 | 1<<7);
+ PORTF |= (1<<4 | 1<<5 | 1<<6 | 1<<7);
+ DDRE &= ~(1<<6);
+ PORTE |= (1<<6);
+ DDRD &= ~(1<<7);
+ PORTD |= (1<<7);
+ DDRB &= ~(1<<1 | 1<<2 | 1<<3 | 1<<4 | 1<<5 | 1<<6);
+ PORTB |= (1<<1 | 1<<2 | 1<<3 | 1<<4 | 1<<5 | 1<<6);
+}
+
+static matrix_row_t read_cols(void)
+{
+ return (PIND&(1<<7) ? 0 : (1<<0)) |
+ (PINE&(1<<6) ? 0 : (1<<1)) |
+ (PINB&(1<<4) ? 0 : (1<<2)) |
+ (PINB&(1<<5) ? 0 : (1<<3)) |
+ (PINB&(1<<6) ? 0 : (1<<4)) |
+ (PINB&(1<<2) ? 0 : (1<<5)) |
+ (PINB&(1<<3) ? 0 : (1<<6)) |
+ (PINB&(1<<1) ? 0 : (1<<7)) |
+ (PINF&(1<<7) ? 0 : (1<<8)) |
+ (PINF&(1<<6) ? 0 : (1<<9)) |
+ (PINF&(1<<5) ? 0 : (1<<10)) |
+ (PINF&(1<<4) ? 0 : (1<<11));
+}
+
+/* Row pin configuration
+ * row: 0 1 2 3
+ * pin: D1 D0 D4 C6
+ */
+
+static void unselect_rows(void)
+{
+ // Hi-Z(DDR:0, PORT:0) to unselect
+ DDRD &= ~0b00010011;
+ PORTD &= ~0b00010011;
+ DDRC &= ~0b01000000;
+ PORTC &= ~0b01000000;
+}
+
+static void select_row(uint8_t row)
+{
+ // Output low(DDR:1, PORT:0) to select
+ switch (row) {
+ case 0:
+ DDRD |= (1<<1);
+ PORTD &= ~(1<<1);
+ break;
+ case 1:
+ DDRD |= (1<<0);
+ PORTD &= ~(1<<0);
+ break;
+ case 2:
+ DDRD |= (1<<4);
+ PORTD &= ~(1<<4);
+ break;
+ case 3:
+ DDRC |= (1<<6);
+ PORTC &= ~(1<<6);
+ break;
+ }
+}
diff --git a/keyboards/ut47/readme.md b/keyboards/ut47/readme.md
new file mode 100644
index 0000000000..3f08b3b329
--- /dev/null
+++ b/keyboards/ut47/readme.md
@@ -0,0 +1,41 @@
+# ut47
+
+![ut47](https://i.imgur.com/ZDKZQaj.jpg)
+
+Somewhere between ortholinear and standard offset. Created to have all the same functions on a Planck in a keyboard but with a more conventional keyboard row stagger.
+
+Keyboard Maintainer: [filoxo](https://github.com/filoxo), [network_operations]
+
+Hardware Supported: [PCB design](http://www.40percent.club/2016/10/gnap-20-plateless.html), Arduino Pro Micro
+
+Hardware Availability: [How to order](http://www.40percent.club/2017/03/ordering-pcb.html)
+
+### Instructions
+
+To flash the UT47 without LEDs (single controller), run:
+
+ make ut47:default
+
+To enable the UT47 LEDs (dual controller), run this for the main controller:
+
+ make ut47:default LED_ENABLE=yes
+
+Or you can add `LED_ENABLE = yes` to *rules.mk*
+
+And then flash [LED_controls.ino](LED_controls.ino) to the second controller using [Arduino IDE](https://www.arduino.cc/en/Main/Software) or similar. NOTE: Arduino IDE will require importing additional libraries to compile.
+
+<small>The reason this is an "opt-in" feature is to prevent sending serial communication over the pin, in case it ends up being used for something else (like RGB underglow).</small>
+
+### Layout
+
+Go to the [default layout README](keymaps/default/readme.md) for more information.
+
+### Additional info
+
+Credit: Forked from [di0ib TMK version](https://github.com/di0ib/tmk_keyboard/tree/master/keyboard/gnap)
+
+See [build environment setup](https://docs.qmk.fm/build_environment_setup.html) then the [make instructions](https://docs.qmk.fm/make_instructions.html) for more information.
+
+### Build Guide
+
+[u/network_operations' build guide thread](https://www.reddit.com/r/MechanicalKeyboards/comments/7wqktu/gnap_the_cheap_40/)
diff --git a/keyboards/ut47/rules.mk b/keyboards/ut47/rules.mk
new file mode 100644
index 0000000000..b046598669
--- /dev/null
+++ b/keyboards/ut47/rules.mk
@@ -0,0 +1,70 @@
+# MCU name
+#MCU = at90usb1286
+MCU = atmega32u4
+
+# Processor frequency.
+# This will define a symbol, F_CPU, in all source code files equal to the
+# processor frequency in Hz. You can then use this symbol in your source code to
+# calculate timings. Do NOT tack on a 'UL' at the end, this will be done
+# automatically to create a 32-bit value in your source code.
+#
+# This will be an integer division of F_USB below, as it is sourced by
+# F_USB after it has run through any CPU prescalers. Note that this value
+# does not *change* the processor frequency - it should merely be updated to
+# reflect the processor speed set externally so that the code can use accurate
+# software delays.
+F_CPU = 16000000
+
+
+#
+# LUFA specific
+#
+# Target architecture (see library "Board Types" documentation).
+ARCH = AVR8
+
+# Input clock frequency.
+# This will define a symbol, F_USB, in all source code files equal to the
+# input clock frequency (before any prescaling is performed) in Hz. This value may
+# differ from F_CPU if prescaling is used on the latter, and is required as the
+# raw input clock is fed directly to the PLL sections of the AVR for high speed
+# clock generation for the USB and other AVR subsections. Do NOT tack on a 'UL'
+# at the end, this will be done automatically to create a 32-bit value in your
+# source code.
+#
+# If no clock division is performed on the input clock inside the AVR (via the
+# CPU clock adjust registers or the clock division fuses), this will be equal to F_CPU.
+F_USB = $(F_CPU)
+
+# Interrupt driven control endpoint task(+60)
+OPT_DEFS += -DINTERRUPT_CONTROL_ENDPOINT
+
+# Bootloader
+BOOTLOADER = caterina
+
+# custom matrix setup
+CUSTOM_MATRIX = yes
+SRC += matrix.c protocol/serial_uart.c
+
+# Build Options
+# change yes to no to disable
+#
+BOOTMAGIC_ENABLE = yes # Virtual DIP switch configuration(+1000)
+MOUSEKEY_ENABLE = yes # Mouse keys(+4700)
+EXTRAKEY_ENABLE = yes # Audio control and System control(+450)
+CONSOLE_ENABLE = no # Console for debug(+400)
+COMMAND_ENABLE = yes # Commands for debug and configuration
+# Do not enable SLEEP_LED_ENABLE. it uses the same timer as BACKLIGHT_ENABLE
+SLEEP_LED_ENABLE = no # Breathing sleep LED during USB suspend
+# if this doesn't work, see here: https://github.com/tmk/tmk_keyboard/wiki/FAQ#nkro-doesnt-work
+NKRO_ENABLE = yes # USB Nkey Rollover
+BACKLIGHT_ENABLE = no # Enable keyboard backlight functionality on B7 by default
+MIDI_ENABLE = no # MIDI support (+2400 to 4200, depending on config)
+UNICODE_ENABLE = no # Unicode
+BLUETOOTH_ENABLE = no # Enable Bluetooth with the Adafruit EZ-Key HID
+AUDIO_ENABLE = no # Audio output on port C6
+FAUXCLICKY_ENABLE = no # Use buzzer to emulate clicky switches
+
+ifeq ($(strip $(LED_ENABLE)), yes)
+ OPT_DEFS += -DLED_ENABLE
+ SRC += led.c
+endif
diff --git a/keyboards/ut47/ut47.c b/keyboards/ut47/ut47.c
new file mode 100644
index 0000000000..f467fd130a
--- /dev/null
+++ b/