Advertisement
LandoRo

Pro Micro SSD1306 ADS1115 5Button Menu FINAL

Apr 3rd, 2025 (edited)
483
0
Never
Not a member of Pastebin yet? Sign Up, it unlocks many cool features!
C# 13.60 KB | Gaming | 0 0
  1. #include <Joystick.h>
  2. #include <Wire.h>
  3. #include <Adafruit_ADS1X15.h>
  4. #include <Adafruit_GFX.h>
  5. #include <Adafruit_SSD1306.h>
  6. #include <EEPROM.h>
  7.  
  8. #define SCREEN_WIDTH 128
  9. #define SCREEN_HEIGHT 32
  10. #define OLED_RESET    -1
  11. #define ADC_MAX 32767
  12.  
  13. // Button pins
  14. #define BUTTON_CLUTCH 5
  15. #define BUTTON_BRAKE 6
  16. #define BUTTON_GAS 4
  17. #define BUTTON_MENU 7
  18. #define BUTTON_HANDBRAKE 8
  19.  
  20. // Output ranges
  21. #define OUT_MIN 0
  22. #define OUT_MAX 32767
  23. #define BRAKE_PSI_MIN 0
  24. #define BRAKE_PSI_MAX 100
  25.  
  26. // Default ADC ranges
  27. const int16_t CLUTCH_DEFAULT_ADC_MIN = 0;
  28. const int16_t CLUTCH_DEFAULT_ADC_MAX = 32767;
  29. const int16_t BRAKE_DEFAULT_ADC_MIN = 4096;
  30. const int16_t BRAKE_DEFAULT_ADC_MAX = 29491;
  31. const int16_t GAS_DEFAULT_ADC_MIN = 0;
  32. const int16_t GAS_DEFAULT_ADC_MAX = 32767;
  33. const int16_t HANDBRAKE_DEFAULT_ADC_MIN = 0;
  34. const int16_t HANDBRAKE_DEFAULT_ADC_MAX = 32767;
  35.  
  36. // Fixed PSI range
  37. const int16_t BRAKE_PSI_ADC_MIN = 4096;
  38. const int16_t BRAKE_PSI_ADC_MAX = 29491;
  39.  
  40. // Kg lookup table (PSI 0–100 → Kg 0–138)
  41. const uint8_t kgTable[101] PROGMEM = {
  42.   0, 1, 3, 4, 6, 7, 8, 10, 11, 12, 14, 15, 17, 18, 19, 21, 22, 23, 25, 26,
  43.   28, 29, 30, 32, 33, 35, 36, 37, 39, 40, 41, 43, 44, 46, 47, 48, 50, 51, 52, 54,
  44.   55, 57, 58, 59, 61, 62, 63, 65, 66, 68, 69, 70, 72, 73, 74, 76, 77, 79, 80, 81,
  45.   83, 84, 85, 87, 88, 90, 91, 92, 94, 95, 96, 98, 99, 101, 102, 103, 105, 106, 107, 109,
  46.   110, 112, 113, 114, 116, 117, 118, 120, 121, 123, 124, 125, 127, 128, 130, 131, 132, 134, 135, 136, 138
  47. };
  48.  
  49. // EEPROM addresses
  50. #define EEPROM_CLUTCH_MIN 0
  51. #define EEPROM_CLUTCH_MAX 2
  52. #define EEPROM_BRAKE_MIN 4
  53. #define EEPROM_BRAKE_MAX 6
  54. #define EEPROM_GAS_MIN 8
  55. #define EEPROM_GAS_MAX 10
  56. #define EEPROM_HANDBRAKE_MIN 12
  57. #define EEPROM_HANDBRAKE_MAX 14
  58. #define EEPROM_INIT_FLAG 16
  59.  
  60. // Enums for states
  61. enum CalibrationStep { NONE = 0, MIN_CAL = 1, MAX_CAL = 2 };
  62. enum SetMessageType { MIN_SET = 1, MAX_SET = 2, DEFAULT_SET = 3 };
  63.  
  64. // Axis data structure
  65. struct Axis {
  66.   int16_t adcMin;
  67.   int16_t adcMax;
  68.   const int16_t defaultMin;
  69.   const int16_t defaultMax;
  70.   bool buttonPressed;
  71.   unsigned long pressStartTime;
  72. };
  73.  
  74. // Global objects and variables
  75. Adafruit_SSD1306 display(SCREEN_WIDTH, SCREEN_HEIGHT, &Wire, OLED_RESET);
  76. Adafruit_ADS1115 ads;
  77. Joystick_ Joystick(JOYSTICK_DEFAULT_REPORT_ID, JOYSTICK_TYPE_JOYSTICK, 0, 0, true, true, true, true, false, false, false, false);
  78. Axis axes[4] = {
  79.   {CLUTCH_DEFAULT_ADC_MIN, CLUTCH_DEFAULT_ADC_MAX, CLUTCH_DEFAULT_ADC_MIN, CLUTCH_DEFAULT_ADC_MAX, false, 0}, // Clutch
  80.   {BRAKE_DEFAULT_ADC_MIN, BRAKE_DEFAULT_ADC_MAX, BRAKE_DEFAULT_ADC_MIN, BRAKE_DEFAULT_ADC_MAX, false, 0},   // Brake
  81.   {GAS_DEFAULT_ADC_MIN, GAS_DEFAULT_ADC_MAX, GAS_DEFAULT_ADC_MIN, GAS_DEFAULT_ADC_MAX, false, 0},           // Gas
  82.   {HANDBRAKE_DEFAULT_ADC_MIN, HANDBRAKE_DEFAULT_ADC_MAX, HANDBRAKE_DEFAULT_ADC_MIN, HANDBRAKE_DEFAULT_ADC_MAX, false, 0} // Hand brake
  83. };
  84. byte calibratingInput = 0;
  85. CalibrationStep calibrationStep = NONE;
  86. unsigned long calibrationStartTime = 0;
  87. bool showingResetMessage = false;
  88. unsigned long resetMessageStartTime = 0;
  89. byte resetMessageInput = 0;
  90. bool showingSetMessage = false;
  91. unsigned long setMessageStartTime = 0;
  92. SetMessageType setMessageType = MIN_SET;
  93. byte setMessageInput = 0;
  94. int16_t setMessageValue = 0;
  95. int16_t lastAdc[4] = {0, 0, 0, 0};
  96. float lastBrakePSI = -1;
  97. uint8_t lastBrakeKg = 0;
  98. byte currentScreen = 0;
  99. bool menuButtonPressed = false;
  100. unsigned long lastMenuPress = 0;
  101. unsigned long lastUpdate = 0;
  102. const unsigned long UPDATE_INTERVAL = 10;
  103.  
  104. void saveCalibrationToEEPROM() {
  105.   EEPROM.put(EEPROM_CLUTCH_MIN, axes[0].adcMin);
  106.   EEPROM.put(EEPROM_CLUTCH_MAX, axes[0].adcMax);
  107.   EEPROM.put(EEPROM_BRAKE_MIN, axes[1].adcMin);
  108.   EEPROM.put(EEPROM_BRAKE_MAX, axes[1].adcMax);
  109.   EEPROM.put(EEPROM_GAS_MIN, axes[2].adcMin);
  110.   EEPROM.put(EEPROM_GAS_MAX, axes[2].adcMax);
  111.   EEPROM.put(EEPROM_HANDBRAKE_MIN, axes[3].adcMin);
  112.   EEPROM.put(EEPROM_HANDBRAKE_MAX, axes[3].adcMax);
  113.   EEPROM.update(EEPROM_INIT_FLAG, 0xAA);
  114. }
  115.  
  116. void loadCalibrationFromEEPROM() {
  117.   if (EEPROM.read(EEPROM_INIT_FLAG) == 0xAA) {
  118.     EEPROM.get(EEPROM_CLUTCH_MIN, axes[0].adcMin);
  119.     EEPROM.get(EEPROM_CLUTCH_MAX, axes[0].adcMax);
  120.     EEPROM.get(EEPROM_BRAKE_MIN, axes[1].adcMin);
  121.     EEPROM.get(EEPROM_BRAKE_MAX, axes[1].adcMax);
  122.     EEPROM.get(EEPROM_GAS_MIN, axes[2].adcMin);
  123.     EEPROM.get(EEPROM_GAS_MAX, axes[2].adcMax);
  124.     EEPROM.get(EEPROM_HANDBRAKE_MIN, axes[3].adcMin);
  125.     EEPROM.get(EEPROM_HANDBRAKE_MAX, axes[3].adcMax);
  126.   } else {
  127.     saveCalibrationToEEPROM();
  128.   }
  129. }
  130.  
  131. void setup() {
  132.   Wire.begin();
  133.   if (!display.begin(SSD1306_SWITCHCAPVCC, 0x3C) || !ads.begin()) while (1);
  134.  
  135.   ads.setGain(GAIN_TWOTHIRDS);
  136.   display.setTextSize(1);
  137.   display.setTextColor(SSD1306_WHITE);
  138.  
  139.   Joystick.begin();
  140.   Joystick.setXAxisRange(OUT_MIN, OUT_MAX);
  141.   Joystick.setYAxisRange(OUT_MIN, OUT_MAX);
  142.   Joystick.setZAxisRange(OUT_MIN, OUT_MAX);
  143.   Joystick.setRxAxisRange(OUT_MIN, OUT_MAX);
  144.  
  145.   // Configure used pins
  146.   pinMode(BUTTON_CLUTCH, INPUT_PULLUP);
  147.   pinMode(BUTTON_BRAKE, INPUT_PULLUP);
  148.   pinMode(BUTTON_GAS, INPUT_PULLUP);
  149.   pinMode(BUTTON_MENU, INPUT_PULLUP);
  150.   pinMode(BUTTON_HANDBRAKE, INPUT_PULLUP);
  151.  
  152.   // Configure unused pins as INPUT_PULLUP
  153.   pinMode(0, INPUT_PULLUP);  // RX
  154.   pinMode(1, INPUT_PULLUP);  // TX
  155.   pinMode(9, INPUT_PULLUP);  // Free pin
  156.   pinMode(10, INPUT_PULLUP); // SS
  157.   pinMode(14, INPUT_PULLUP); // MISO
  158.   pinMode(15, INPUT_PULLUP); // SCK
  159.   pinMode(16, INPUT_PULLUP); // MOSI
  160.   pinMode(18, INPUT_PULLUP); // A0
  161.   pinMode(19, INPUT_PULLUP); // A1
  162.   pinMode(20, INPUT_PULLUP); // A2
  163.   pinMode(21, INPUT_PULLUP); // A3
  164.  
  165.   loadCalibrationFromEEPROM();
  166. }
  167.  
  168. void handleButton(byte index, int pin) {
  169.   if (digitalRead(pin) == LOW) {
  170.     if (!axes[index].buttonPressed) {
  171.       axes[index].buttonPressed = true;
  172.       axes[index].pressStartTime = millis();
  173.     } else if (millis() - axes[index].pressStartTime >= 3000) {
  174.       axes[index].adcMin = axes[index].defaultMin;
  175.       axes[index].adcMax = axes[index].defaultMax;
  176.       saveCalibrationToEEPROM();
  177.       showingResetMessage = true;
  178.       resetMessageStartTime = millis();
  179.       resetMessageInput = index + 1;
  180.       axes[index].buttonPressed = false;
  181.     }
  182.   } else if (axes[index].buttonPressed) {
  183.     axes[index].buttonPressed = false;
  184.     if (millis() - axes[index].pressStartTime < 3000) {
  185.       calibratingInput = index + 1;
  186.       calibrationStep = MIN_CAL;
  187.       calibrationStartTime = millis();
  188.     }
  189.   }
  190. }
  191.  
  192. void handleMenuButton() {
  193.   if (digitalRead(BUTTON_MENU) == LOW) {
  194.     if (!menuButtonPressed && millis() - lastMenuPress > 200) {
  195.       menuButtonPressed = true;
  196.       lastMenuPress = millis();
  197.       currentScreen = (currentScreen + 1) % 3;
  198.     }
  199.   } else {
  200.     menuButtonPressed = false;
  201.   }
  202. }
  203.  
  204. int16_t mapInt(int16_t x, int16_t in_min, int16_t in_max, int16_t out_min, int16_t out_max) {
  205.   return (int16_t)(((long)(x - in_min) * (out_max - out_min)) / (in_max - in_min) + out_min);
  206. }
  207.  
  208. void updateJoystick() {
  209.   lastAdc[0] = ads.readADC_SingleEnded(0); // Clutch
  210.   lastAdc[1] = ads.readADC_SingleEnded(1); // Brake
  211.   lastAdc[2] = ads.readADC_SingleEnded(2); // Gas
  212.   lastAdc[3] = ads.readADC_SingleEnded(3); // Hand brake
  213.  
  214.   Joystick.setXAxis(mapInt(constrain(lastAdc[0], axes[0].adcMin, axes[0].adcMax), axes[0].adcMin, axes[0].adcMax, OUT_MIN, OUT_MAX));
  215.   Joystick.setYAxis(mapInt(constrain(lastAdc[1], axes[1].adcMin, axes[1].adcMax), axes[1].adcMin, axes[1].adcMax, OUT_MIN, OUT_MAX));
  216.   Joystick.setZAxis(mapInt(constrain(lastAdc[2], axes[2].adcMin, axes[2].adcMax), axes[2].adcMin, axes[2].adcMax, OUT_MIN, OUT_MAX));
  217.   Joystick.setRxAxis(mapInt(constrain(lastAdc[3], axes[3].adcMin, axes[3].adcMax), axes[3].adcMin, axes[3].adcMax, OUT_MIN, OUT_MAX));
  218.   lastBrakePSI = mapFloat(constrain(lastAdc[1], BRAKE_PSI_ADC_MIN, BRAKE_PSI_ADC_MAX), BRAKE_PSI_ADC_MIN, BRAKE_PSI_ADC_MAX, BRAKE_PSI_MIN, BRAKE_PSI_MAX);
  219.   lastBrakeKg = pgm_read_byte(&kgTable[(uint8_t)lastBrakePSI]);
  220. }
  221.  
  222. void updateDisplay() {
  223.   display.clearDisplay();
  224.   display.setCursor(0, 0);
  225.  
  226.   if (calibratingInput != 0 && (calibrationStep == MIN_CAL || calibrationStep == MAX_CAL)) {
  227.     unsigned long elapsed = millis() - calibrationStartTime;
  228.     int secondsLeft = 5 - (elapsed / 1000);
  229.     if (secondsLeft < 0) secondsLeft = 0;
  230.  
  231.     display.print(F("Hold "));
  232.     switch (calibratingInput) {
  233.       case 1: display.print(F("Clutch")); break;
  234.       case 2: display.print(F("Brake")); break;
  235.       case 3: display.print(F("Gas")); break;
  236.       case 4: display.print(F("Handbrake")); break;
  237.     }
  238.     display.print(calibrationStep == MIN_CAL ? F(" MIN: ") : F(" MAX: "));
  239.     display.print(secondsLeft);
  240.     display.setCursor(0, 10);
  241.     display.print(F("ADC: "));
  242.     display.print(lastAdc[calibratingInput - 1]);
  243.  
  244.     if (elapsed >= 5000) {
  245.       int16_t currentADC = lastAdc[calibratingInput - 1];
  246.       if (calibrationStep == MIN_CAL) {
  247.         axes[calibratingInput - 1].adcMin = currentADC;
  248.         saveCalibrationToEEPROM();
  249.         showingSetMessage = true;
  250.         setMessageStartTime = millis();
  251.         setMessageType = MIN_SET;
  252.         setMessageInput = calibratingInput;
  253.         setMessageValue = currentADC;
  254.         calibrationStep = NONE;
  255.       } else if (calibrationStep == MAX_CAL) {
  256.         axes[calibratingInput - 1].adcMax = currentADC;
  257.         saveCalibrationToEEPROM();
  258.         showingSetMessage = true;
  259.         setMessageStartTime = millis();
  260.         setMessageType = MAX_SET;
  261.         setMessageInput = calibratingInput;
  262.         setMessageValue = currentADC;
  263.         calibratingInput = 0;
  264.         calibrationStep = NONE;
  265.         showingResetMessage = false;
  266.       }
  267.     }
  268.   } else if (showingSetMessage && millis() - setMessageStartTime < 2000) {
  269.     display.print(F("DONE"));
  270.   } else if (showingResetMessage && millis() - resetMessageStartTime < 3000) {
  271.     switch (resetMessageInput) {
  272.       case 1: display.print(F("Clutch")); break;
  273.       case 2: display.print(F("Brake")); break;
  274.       case 3: display.print(F("Gas")); break;
  275.       case 4: display.print(F("Handbrake")); break;
  276.     }
  277.     display.print(F(" reset to defaults"));
  278.   } else if (showingResetMessage && millis() - resetMessageStartTime >= 3000) {
  279.     showingResetMessage = false;
  280.     showingSetMessage = true;
  281.     setMessageStartTime = millis();
  282.     setMessageType = DEFAULT_SET;
  283.     setMessageInput = resetMessageInput;
  284.     setMessageValue = 0;
  285.   } else {
  286.     if (showingSetMessage && millis() - setMessageStartTime >= 2000) {
  287.       showingSetMessage = false;
  288.       setMessageInput = 0;
  289.       setMessageValue = 0;
  290.       if (setMessageType == MIN_SET && calibratingInput != 0) {
  291.         calibrationStep = MAX_CAL;
  292.         calibrationStartTime = millis();
  293.       }
  294.     }
  295.  
  296.     if (currentScreen == 0) { // Screen 1: Min/Max/ADC
  297.       display.print(F("C:"));
  298.       display.print(axes[0].adcMin);
  299.       display.print(F(" "));
  300.       display.print(axes[0].adcMax);
  301.       display.print(F(" "));
  302.       display.print(lastAdc[0]);
  303.       display.setCursor(0, 8);
  304.       display.print(F("B:"));
  305.       display.print(axes[1].adcMin);
  306.       display.print(F(" "));
  307.       display.print(axes[1].adcMax);
  308.       display.print(F(" "));
  309.       display.print(lastAdc[1]);
  310.       display.setCursor(0, 16);
  311.       display.print(F("G:"));
  312.       display.print(axes[2].adcMin);
  313.       display.print(F(" "));
  314.       display.print(axes[2].adcMax);
  315.       display.print(F(" "));
  316.       display.print(lastAdc[2]);
  317.       display.setCursor(0, 24);
  318.       display.print(F("H:"));
  319.       display.print(axes[3].adcMin);
  320.       display.print(F(" "));
  321.       display.print(axes[3].adcMax);
  322.       display.print(F(" "));
  323.       display.print(lastAdc[3]);
  324.     } else if (currentScreen == 1) { // Screen 2: Brake PSI and Kg
  325.       display.print(F("Brake PSI: "));
  326.       display.print(lastBrakePSI, 2);
  327.       display.setCursor(0, 8);
  328.       display.print(F("Kg: "));
  329.       display.print(lastBrakeKg);
  330.     } else if (currentScreen == 2) { // Screen 3: Clutch/Brake/Gas/Handbrake Percentages
  331.       int16_t clutchPercent = mapInt(constrain(lastAdc[0], axes[0].adcMin, axes[0].adcMax), axes[0].adcMin, axes[0].adcMax, 0, 100);
  332.       int16_t brakePercent = mapInt(constrain(lastAdc[1], axes[1].adcMin, axes[1].adcMax), axes[1].adcMin, axes[1].adcMax, 0, 100);
  333.       int16_t gasPercent = mapInt(constrain(lastAdc[2], axes[2].adcMin, axes[2].adcMax), axes[2].adcMin, axes[2].adcMax, 0, 100);
  334.       int16_t handbrakePercent = mapInt(constrain(lastAdc[3], axes[3].adcMin, axes[3].adcMax), axes[3].adcMin, axes[3].adcMax, 0, 100);
  335.       display.print(F("C:"));
  336.       display.print(clutchPercent);
  337.       display.print(F("%"));
  338.       display.setCursor(0, 8);
  339.       display.print(F("B:"));
  340.       display.print(brakePercent);
  341.       display.print(F("%"));
  342.       display.setCursor(0, 16);
  343.       display.print(F("G:"));
  344.       display.print(gasPercent);
  345.       display.print(F("%"));
  346.       display.setCursor(0, 24);
  347.       display.print(F("H:"));
  348.       display.print(handbrakePercent);
  349.       display.print(F("%"));
  350.     }
  351.   }
  352.   display.display();
  353. }
  354.  
  355. void loop() {
  356.   unsigned long currentMillis = millis();
  357.  
  358.   if (currentMillis - lastUpdate >= UPDATE_INTERVAL) {
  359.     if (calibratingInput == 0 && !showingResetMessage && !showingSetMessage) {
  360.       handleButton(0, BUTTON_CLUTCH);
  361.       handleButton(1, BUTTON_BRAKE);
  362.       handleButton(2, BUTTON_GAS);
  363.       handleButton(3, BUTTON_HANDBRAKE);
  364.     }
  365.     handleMenuButton();
  366.     updateJoystick();
  367.     updateDisplay();
  368.     lastUpdate = currentMillis;
  369.   }
  370. }
  371.  
  372. inline float mapFloat(float x, float in_min, float in_max, float out_min, float out_max) {
  373.   return (x - in_min) * (out_max - out_min) / (in_max - in_min) + out_min;
  374. }
Tags: sim racing
Advertisement
Add Comment
Please, Sign In to add comment
Advertisement