diff --git a/Marlin/Configuration.h b/Marlin/Configuration.h
index 60ca46744df4..7978cbc05525 100644
--- a/Marlin/Configuration.h
+++ b/Marlin/Configuration.h
@@ -2939,23 +2939,16 @@
/**
* DGUS Touch Display with DWIN OS. (Choose one.)
- * ORIGIN : https://www.aliexpress.com/item/32993409517.html
- * FYSETC : https://www.aliexpress.com/item/32961471929.html
- * MKS : https://www.aliexpress.com/item/1005002008179262.html
- *
- * Flash display with DGUS Displays for Marlin:
- * - Format the SD card to FAT32 with an allocation size of 4kb.
- * - Download files as specified for your type of display.
- * - Plug the microSD card into the back of the display.
- * - Boot the display and wait for the update to complete.
*
* ORIGIN (Marlin DWIN_SET)
* - Download https://github.com/coldtobi/Marlin_DGUS_Resources
* - Copy the downloaded DWIN_SET folder to the SD card.
+ * - Product: https://www.aliexpress.com/item/32993409517.html
*
* FYSETC (Supplier default)
* - Download https://github.com/FYSETC/FYSTLCD-2.0
* - Copy the downloaded SCREEN folder to the SD card.
+ * - Product: https://www.aliexpress.com/item/32961471929.html
*
* HIPRECY (Supplier default)
* - Download https://github.com/HiPrecy/Touch-Lcd-LEO
@@ -2964,17 +2957,24 @@
* MKS (MKS-H43) (Supplier default)
* - Download https://github.com/makerbase-mks/MKS-H43
* - Copy the downloaded DWIN_SET folder to the SD card.
+ * - Product: https://www.aliexpress.com/item/1005002008179262.html
*
* RELOADED (T5UID1)
* - Download https://github.com/Desuuuu/DGUS-reloaded/releases
* - Copy the downloaded DWIN_SET folder to the SD card.
+ *
+ * IA_CREALITY (T5UID1)
+ * - Download https://github.com/InsanityAutomation/Marlin/raw/CrealityDwin2.0_Bleeding/TM3D_Combined480272_Landscape_V7.7z
+ * - Copy the downloaded DWIN_SET folder to the SD card.
+ *
+ * Flash display with DGUS Displays for Marlin:
+ * - Format the SD card to FAT32 with an allocation size of 4kb.
+ * - Download files as specified for your type of display.
+ * - Plug the microSD card into the back of the display.
+ * - Boot the display and wait for the update to complete.
*/
-//#define DGUS_LCD_UI_ORIGIN
-//#define DGUS_LCD_UI_FYSETC
-//#define DGUS_LCD_UI_HIPRECY
-//#define DGUS_LCD_UI_MKS
-//#define DGUS_LCD_UI_RELOADED
-#if ENABLED(DGUS_LCD_UI_MKS)
+//#define DGUS_LCD_UI ORIGIN
+#if DGUS_UI_IS(MKS)
#define USE_MKS_GREEN_UI
#endif
diff --git a/Marlin/Configuration_adv.h b/Marlin/Configuration_adv.h
index 18fe3d71429c..7b80dfa76b5f 100644
--- a/Marlin/Configuration_adv.h
+++ b/Marlin/Configuration_adv.h
@@ -1864,11 +1864,11 @@
#define DGUS_UPDATE_INTERVAL_MS 500 // (ms) Interval between automatic screen updates
- #if ANY(DGUS_LCD_UI_FYSETC, DGUS_LCD_UI_MKS, DGUS_LCD_UI_HIPRECY)
+ #if DGUS_UI_IS(FYSETC, MKS, HIPRECY)
#define DGUS_PRINT_FILENAME // Display the filename during printing
#define DGUS_PREHEAT_UI // Display a preheat screen during heatup
- #if EITHER(DGUS_LCD_UI_FYSETC, DGUS_LCD_UI_MKS)
+ #if DGUS_UI_IS(FYSETC, MKS)
//#define DGUS_UI_MOVE_DIS_OPTION // Disabled by default for FYSETC and MKS
#else
#define DGUS_UI_MOVE_DIS_OPTION // Enabled by default for UI_HIPRECY
diff --git a/Marlin/src/core/macros.h b/Marlin/src/core/macros.h
index 3029009c06b8..ca736ba9a981 100644
--- a/Marlin/src/core/macros.h
+++ b/Marlin/src/core/macros.h
@@ -742,3 +742,13 @@
#define _HAS_E_TEMP(N) || TEMP_SENSOR(N)
#define HAS_E_TEMP_SENSOR (0 REPEAT(EXTRUDERS, _HAS_E_TEMP))
#define TEMP_SENSOR_IS_MAX_TC(T) (TEMP_SENSOR(T) == -5 || TEMP_SENSOR(T) == -3 || TEMP_SENSOR(T) == -2)
+
+#define _UI_NONE 0
+#define _UI_ORIGIN 101
+#define _UI_FYSETC 102
+#define _UI_HIPRECY 103
+#define _UI_MKS 104
+#define _UI_RELOADED 105
+#define _UI_IA_CREALITY 106
+#define _DGUS_UI_IS(N) || (DGUS_LCD_UI == _UI_ ## N)
+#define DGUS_UI_IS(V...) (0 MAP(_DGUS_UI_IS, V))
diff --git a/Marlin/src/inc/Conditionals_LCD.h b/Marlin/src/inc/Conditionals_LCD.h
index 545d55349475..e30de873048c 100644
--- a/Marlin/src/inc/Conditionals_LCD.h
+++ b/Marlin/src/inc/Conditionals_LCD.h
@@ -36,6 +36,21 @@
#define FYSETC_MINI_12864_2_1
#endif
+// Updated DGUS_UI shorthand single option can be used, or old settings, for now
+#if DGUS_UI_IS(ORIGIN)
+ #define DGUS_LCD_UI_ORIGIN
+#elif DGUS_UI_IS(FYSETC)
+ #define DGUS_LCD_UI_FYSETC
+#elif DGUS_UI_IS(HIPRECY)
+ #define DGUS_LCD_UI_HIPRECY
+#elif DGUS_UI_IS(MKS)
+ #define DGUS_LCD_UI_MKS
+#elif DGUS_UI_IS(RELOADED)
+ #define DGUS_LCD_UI_RELOADED
+#elif DGUS_UI_IS(IA_CREALITY)
+ #define DGUS_LCD_UI_IA_CREALITY
+#endif
+
/**
* General Flags that may be set below by specific LCDs
*
@@ -462,12 +477,11 @@
#endif
// Aliases for LCD features
-#if ANY(DGUS_LCD_UI_ORIGIN, DGUS_LCD_UI_FYSETC, DGUS_LCD_UI_HIPRECY, DGUS_LCD_UI_MKS)
- #define HAS_DGUS_LCD_CLASSIC 1
-#endif
-
-#if EITHER(HAS_DGUS_LCD_CLASSIC, DGUS_LCD_UI_RELOADED)
+#if !DGUS_UI_IS(NONE)
#define HAS_DGUS_LCD 1
+ #if DGUS_UI_IS(ORIGIN, FYSETC, HIPRECY, MKS)
+ #define HAS_DGUS_LCD_CLASSIC 1
+ #endif
#endif
// Extensible UI serial touch screens. (See src/lcd/extui)
diff --git a/Marlin/src/inc/SanityCheck.h b/Marlin/src/inc/SanityCheck.h
index 79306c77a6ae..ca87e87a343b 100644
--- a/Marlin/src/inc/SanityCheck.h
+++ b/Marlin/src/inc/SanityCheck.h
@@ -513,7 +513,7 @@
#elif defined(HOME_USING_SPREADCYCLE)
#error "HOME_USING_SPREADCYCLE is now obsolete. Please remove it."
#elif defined(DGUS_LCD)
- #error "DGUS_LCD is now DGUS_LCD_UI_(ORIGIN|FYSETC|HIPRECY)."
+ #error "DGUS_LCD is now DGUS_LCD_UI ORIGIN|FYSETC|HIPRECY)."
#elif defined(DGUS_SERIAL_PORT)
#error "DGUS_SERIAL_PORT is now LCD_SERIAL_PORT."
#elif defined(DGUS_BAUDRATE)
@@ -3009,7 +3009,7 @@ static_assert(Y_MAX_LENGTH >= Y_BED_SIZE, "Movement bounds (Y_MIN_POS, Y_MAX_POS
+ (DISABLED(IS_LEGACY_TFT) && ENABLED(TFT_GENERIC)) \
+ (ENABLED(IS_LEGACY_TFT) && COUNT_ENABLED(TFT_320x240, TFT_320x240_SPI, TFT_480x320, TFT_480x320_SPI)) \
+ COUNT_ENABLED(ANYCUBIC_LCD_I3MEGA, ANYCUBIC_LCD_CHIRON, ANYCUBIC_TFT35) \
- + COUNT_ENABLED(DGUS_LCD_UI_ORIGIN, DGUS_LCD_UI_FYSETC, DGUS_LCD_UI_HIPRECY, DGUS_LCD_UI_MKS, DGUS_LCD_UI_RELOADED) \
+ + DGUS_UI_IS(ORIGIN) + DGUS_UI_IS(FYSETC) + DGUS_UI_IS(HIPRECY) + DGUS_UI_IS(MKS) + DGUS_UI_IS(RELOADED) + DGUS_UI_IS(IA_CREALITY) \
+ COUNT_ENABLED(ENDER2_STOCKDISPLAY, CR10_STOCKDISPLAY) \
+ COUNT_ENABLED(DWIN_CREALITY_LCD, DWIN_LCD_PROUI, DWIN_CREALITY_LCD_JYERSUI, DWIN_MARLINUI_PORTRAIT, DWIN_MARLINUI_LANDSCAPE) \
+ COUNT_ENABLED(FYSETC_MINI_12864_X_X, FYSETC_MINI_12864_1_2, FYSETC_MINI_12864_2_0, FYSETC_GENERIC_12864_1_1) \
@@ -4270,34 +4270,51 @@ static_assert(_PLUS_TEST(4), "HOMING_FEEDRATE_MM_M values must be positive.");
#undef _BAD_DRIVER
/**
- * Require certain features for DGUS_LCD_UI_RELOADED.
+ * Require certain features for DGUS_LCD_UI RELOADED.
*/
-#if ENABLED(DGUS_LCD_UI_RELOADED)
+#if DGUS_UI_IS(RELOADED)
#if BUFSIZE < 4
- #error "DGUS_LCD_UI_RELOADED requires a BUFSIZE of at least 4."
+ #error "DGUS_LCD_UI RELOADED requires a BUFSIZE of at least 4."
#elif HOTENDS < 1
- #error "DGUS_LCD_UI_RELOADED requires at least 1 hotend."
+ #error "DGUS_LCD_UI RELOADED requires at least 1 hotend."
#elif EXTRUDERS < 1
- #error "DGUS_LCD_UI_RELOADED requires at least 1 extruder."
+ #error "DGUS_LCD_UI RELOADED requires at least 1 extruder."
#elif !HAS_HEATED_BED
- #error "DGUS_LCD_UI_RELOADED requires a heated bed."
+ #error "DGUS_LCD_UI RELOADED requires a heated bed."
#elif FAN_COUNT < 1
- #error "DGUS_LCD_UI_RELOADED requires a fan."
+ #error "DGUS_LCD_UI RELOADED requires a fan."
#elif !HAS_BED_PROBE
- #error "DGUS_LCD_UI_RELOADED requires a bed probe."
+ #error "DGUS_LCD_UI RELOADED requires a bed probe."
#elif !HAS_MESH
- #error "DGUS_LCD_UI_RELOADED requires mesh leveling."
+ #error "DGUS_LCD_UI RELOADED requires mesh leveling."
#elif DISABLED(LCD_BED_TRAMMING)
- #error "DGUS_LCD_UI_RELOADED requires LCD_BED_TRAMMING."
+ #error "DGUS_LCD_UI RELOADED requires LCD_BED_TRAMMING."
#elif DISABLED(BABYSTEP_ALWAYS_AVAILABLE)
- #error "DGUS_LCD_UI_RELOADED requires BABYSTEP_ALWAYS_AVAILABLE."
+ #error "DGUS_LCD_UI RELOADED requires BABYSTEP_ALWAYS_AVAILABLE."
#elif DISABLED(BABYSTEP_ZPROBE_OFFSET)
- #error "DGUS_LCD_UI_RELOADED requires BABYSTEP_ZPROBE_OFFSET."
+ #error "DGUS_LCD_UI RELOADED requires BABYSTEP_ZPROBE_OFFSET."
#elif ENABLED(AUTO_BED_LEVELING_UBL) && DISABLED(UBL_SAVE_ACTIVE_ON_M500)
#warning "Without UBL_SAVE_ACTIVE_ON_M500, your mesh will not be saved when using the touchscreen."
#endif
#endif
+/**
+ * Require certain features for DGUS_LCD_UI IA_CREALITY.
+ */
+#if DGUS_UI_IS(IA_CREALITY)
+ #if DISABLED(ADVANCED_PAUSE_FEATURE)
+ #error "DGUS_LCD_UI IA_CREALITY requires ADVANCED_PAUSE_FEATURE."
+ #elif DISABLED(LCD_BED_TRAMMING)
+ #error "DGUS_LCD_UI IA_CREALITY requires LCD_BED_TRAMMING."
+ #elif DISABLED(CLASSIC_JERK)
+ #error "DGUS_LCD_UI IA_CREALITY requires CLASSIC_JERK."
+ #elif DISABLED(BABYSTEPPING)
+ #error "DGUS_LCD_UI IA_CREALITY requires BABYSTEPPING."
+ #elif NONE(AUTO_BED_LEVELING_BILINEAR, AUTO_BED_LEVELING_UBL, MESH_BED_LEVELING)
+ #error "DGUS_LCD_UI IA_CREALITY requires AUTO_BED_LEVELING_BILINEAR, AUTO_BED_LEVELING_UBL, or MESH_BED_LEVELING."
+ #endif
+#endif
+
// JTAG support in the HAL
#if ENABLED(DISABLE_DEBUG) && !defined(JTAGSWD_DISABLE)
#error "DISABLE_DEBUG is not supported for the selected MCU/Board."
diff --git a/Marlin/src/lcd/extui/ia_creality/FileNavigator.cpp b/Marlin/src/lcd/extui/ia_creality/FileNavigator.cpp
new file mode 100644
index 000000000000..7a191e906ac6
--- /dev/null
+++ b/Marlin/src/lcd/extui/ia_creality/FileNavigator.cpp
@@ -0,0 +1,183 @@
+/**
+ * Marlin 3D Printer Firmware
+ * Copyright (c) 2022 MarlinFirmware [https://github.com/MarlinFirmware/Marlin]
+ *
+ * Based on Sprinter and grbl.
+ * Copyright (c) 2011 Camiel Gubbels / Erik van der Zalm
+ *
+ * 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 3 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 .
+ *
+ */
+
+/* ****************************************
+ * lcd/extui/ia_creality/FileNavigator.cpp
+ * ****************************************
+ * Extensible_UI implementation for Creality DWIN
+ * 10SPro, Max, CR10V2
+ * Based on implementations for Anycubic Chiron and Nextion by Nick Wells and Skorpi08
+ * Written by Insanity Automation
+ * ***************************************/
+
+#include "../../../inc/MarlinConfigPre.h"
+
+#if ENABLED(DGUS_LCD_UI_IA_CREALITY)
+
+#include "FileNavigator.h"
+
+using namespace ExtUI;
+
+#define DEBUG_OUT ENABLED(DEBUG_DWIN)
+#include "../../../core/debug_out.h"
+
+FileList FileNavigator::filelist; // Instance of the Marlin file API
+char FileNavigator::currentfoldername[MAX_PATH_LEN]; // Current folder path
+uint16_t FileNavigator::lastindex;
+uint8_t FileNavigator::folderdepth;
+uint16_t FileNavigator::currentindex; // override the panel request
+
+FileNavigator filenavigator;
+
+FileNavigator::FileNavigator() { reset(); }
+
+void FileNavigator::reset() {
+ currentfoldername[0] = '\0';
+ folderdepth = 0;
+ currentindex = 0;
+ lastindex = 0;
+ // Start at root folder
+ while (!filelist.isAtRootDir()) filelist.upDir();
+ refresh();
+}
+
+void FileNavigator::refresh() { filelist.refresh(); }
+
+bool FileNavigator::getIndexisDir(uint16_t index){
+ filelist.seek(index);
+ return filelist.isDir();
+}
+
+const char *FileNavigator::getIndexName(uint16_t index){
+ filelist.seek(index);
+ return filelist.shortFilename();
+}
+
+uint16_t FileNavigator::maxFiles() {
+ return filelist.count();
+}
+
+void FileNavigator::getFiles(uint16_t index) {
+ uint16_t files = DISPLAY_FILES, fcnt = 0;
+ if (index == 0)
+ currentindex = 0;
+ else {
+ // Each time we change folder we reset the file index to 0 and keep track
+ // of the current position as the TFT panel isn't aware of folder trees.
+ --currentindex; // go back a file to take account of the .. added to the root.
+ if (index > lastindex)
+ currentindex += files + 1;
+ else if (currentindex >= files)
+ currentindex -= files - 1;
+ else
+ currentindex = 0;
+ }
+ lastindex = index;
+
+
+ // Clear currently drawn screen
+ for (int i = 0; i < DISPLAY_FILES; i++) {
+ for (int j = 0; j < 20; j++)
+ rtscheck.RTS_SndData(0, SDFILE_ADDR + (i * 20) + j);
+ }
+
+ for (int j = 0; j < 10; j++) {
+ rtscheck.RTS_SndData(0, Printfilename + j); // clear screen.
+ rtscheck.RTS_SndData(0, Choosefilename + j); // clear filename
+ }
+ for (int j = 0; j < 8; j++)
+ rtscheck.RTS_SndData(0, FilenameCount + j);
+ for (int j = 1; j <= DISPLAY_FILES; j++) {
+ rtscheck.RTS_SndData(10, FilenameIcon + j);
+ rtscheck.RTS_SndData(10, FilenameIcon1 + j);
+ }
+
+ DEBUG_ECHOLNPGM("index=", index, " currentindex=", currentindex, "folderdepth=", folderdepth);
+
+ if (currentindex == 0 && folderdepth > 0) { // Add a link to go up a folder
+ files--;
+ rtscheck.RTS_SndData("Up Directory", SDFILE_ADDR);
+ fcnt++;
+ }
+ else if (currentindex == DISPLAY_FILES && folderdepth > 0)
+ currentindex--;
+
+ for (uint16_t seek = currentindex; seek < currentindex + files; seek++) {
+ if (filelist.seek(seek)) {
+ const int filelen = strlen(filelist.filename());
+ if (filelen > 20) {
+ char *buf = (char *)filelist.filename();
+ //char buf[filelen];
+ //strcpy(&buf[filelen], filelist.filename());
+ buf[18] = '\0'; // cutoff at screen edge
+ rtscheck.RTS_SndData(buf, (SDFILE_ADDR + (fcnt * 20)));
+ }
+ else
+ rtscheck.RTS_SndData(filelist.filename(), (SDFILE_ADDR + (fcnt * 20)));
+
+ if (filelist.isDir()) {
+ rtscheck.RTS_SndData((uint8_t)4, FilenameIcon + (fcnt+1));
+ rtscheck.RTS_SndData((unsigned long)0x041F, (FilenameNature + ((1+fcnt) * 16))); // Change BG of selected line to Blue
+ }
+ else {
+ rtscheck.RTS_SndData((uint8_t)0, FilenameIcon + (fcnt+1));
+ rtscheck.RTS_SndData((unsigned long)0xFFFF, (FilenameNature + ((1+fcnt) * 16))); // white
+ }
+ SERIAL_ECHOLNPGM("-", seek, " '", filelist.filename(), "' '", currentfoldername, "", filelist.shortFilename(), "'\n");
+ fcnt++;
+ }
+ }
+}
+
+void FileNavigator::changeDIR(char *folder) {
+ DEBUG_ECHOLNPGM("currentfolder: ", currentfoldername, " New: ", folder);
+ if (folderdepth >= MAX_FOLDER_DEPTH) return; // limit the folder depth
+ strcat(currentfoldername, folder);
+ strcat(currentfoldername, "/");
+ filelist.changeDir(folder);
+ refresh();
+ folderdepth++;
+ currentindex = 0;
+}
+
+void FileNavigator::upDIR() {
+ filelist.upDir();
+ refresh();
+ folderdepth--;
+ currentindex = 0;
+ // Remove the last child folder from the stored path
+ if (folderdepth == 0) {
+ currentfoldername[0] = '\0';
+ reset();
+ }
+ else {
+ char *pos = nullptr;
+ for (uint8_t f = 0; f < folderdepth; f++)
+ pos = strchr(currentfoldername, '/');
+ pos[1] = '\0';
+ }
+ DEBUG_ECHOLNPGM("depth: ", folderdepth, " currentfoldername: ", currentfoldername);
+}
+
+char* FileNavigator::getCurrentFolderName() { return currentfoldername; }
+
+#endif // DGUS_LCD_UI_IA_CREALITY
diff --git a/Marlin/src/lcd/extui/ia_creality/FileNavigator.h b/Marlin/src/lcd/extui/ia_creality/FileNavigator.h
new file mode 100644
index 000000000000..29f562845560
--- /dev/null
+++ b/Marlin/src/lcd/extui/ia_creality/FileNavigator.h
@@ -0,0 +1,63 @@
+/**
+ * Marlin 3D Printer Firmware
+ * Copyright (c) 2022 MarlinFirmware [https://github.com/MarlinFirmware/Marlin]
+ *
+ * Based on Sprinter and grbl.
+ * Copyright (c) 2011 Camiel Gubbels / Erik van der Zalm
+ *
+ * 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 3 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 .
+ *
+ */
+#pragma once
+
+/* ****************************************
+ * lcd/extui/ia_creality/FileNavigator.h
+ * ****************************************
+ * Extensible_UI implementation for Creality DWIN
+ * 10SPro, Max, CRX and others
+ * Based on implementations for Anycubic Chiron and Nextion by Nick Wells and Skorpi08
+ * Written by Insanity Automation
+ * ***************************************/
+
+#include "creality_extui.h"
+#include "../ui_api.h"
+
+#define MAX_FOLDER_DEPTH 4 // Limit folder depth TFT has a limit for the file path
+#define MAX_CMND_LEN 16 * MAX_FOLDER_DEPTH // Maximum Length for a Panel command
+#define MAX_PATH_LEN 16 * MAX_FOLDER_DEPTH // Maximum number of characters in a SD file path
+#define DISPLAY_FILES 4
+
+using namespace ExtUI;
+
+class FileNavigator {
+ public:
+ FileNavigator();
+ static void reset();
+ static void getFiles(uint16_t);
+ static void upDIR();
+ static void changeDIR(char *);
+ static void refresh();
+ static char* getCurrentFolderName();
+ static uint8_t folderdepth;
+ static uint16_t currentindex;
+ static bool getIndexisDir(uint16_t);
+ const char *getIndexName(uint16_t);
+ static uint16_t maxFiles();
+ private:
+ static FileList filelist;
+ static char currentfoldername[MAX_PATH_LEN];
+ static uint16_t lastindex;
+};
+
+extern FileNavigator filenavigator;
diff --git a/Marlin/src/lcd/extui/ia_creality/creality_extui.cpp b/Marlin/src/lcd/extui/ia_creality/creality_extui.cpp
new file mode 100644
index 000000000000..4aa7bd8dcdf4
--- /dev/null
+++ b/Marlin/src/lcd/extui/ia_creality/creality_extui.cpp
@@ -0,0 +1,2369 @@
+/**
+ * Marlin 3D Printer Firmware
+ * Copyright (c) 2022 MarlinFirmware [https://github.com/MarlinFirmware/Marlin]
+ *
+ * Based on Sprinter and grbl.
+ * Copyright (c) 2011 Camiel Gubbels / Erik van der Zalm
+ *
+ * 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 3 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 .
+ *
+ */
+
+/* ****************************************
+ * lcd/extui/ia_creality/creality_extui.cpp
+ * ****************************************
+ * Extensible_UI implementation for Creality DWIN
+ * 10SPro, Max, CRX, and others
+ * Based original Creality release, ported to ExtUI for Marlin 2.0
+ * Written by Insanity Automation, sponsored by Tiny Machines 3D
+ *
+ * ***************************************/
+
+#include "../../../inc/MarlinConfigPre.h"
+
+#if ENABLED(DGUS_LCD_UI_IA_CREALITY)
+
+#include "creality_extui.h"
+#include "FileNavigator.h"
+#include "../ui_api.h"
+
+#include
+#include
+#include
+
+#define DEBUG_OUT ENABLED(DEBUG_DWIN)
+#include "../../../core/debug_out.h"
+
+namespace ExtUI {
+ static uint16_t fileIndex = 0;
+ uint8_t recordcount = 0;
+ uint8_t waitway_lock = 0;
+ const float manual_feedrate_mm_m[] = MANUAL_FEEDRATE;
+ uint8_t startprogress = 0;
+
+ char waitway = 0;
+ int recnum = 0;
+ unsigned char Percentrecord = 0;
+ float ChangeMaterialbuf[2] = {0};
+
+ char NozzleTempStatus[3] = {0};
+
+ char PrinterStatusKey[2] = {0}; // PrinterStatusKey[1] value: 0 represents to keep temperature, 1 represents to heating , 2 stands for cooling , 3 stands for printing
+ // PrinterStatusKey[0] value: 0 reprensents 3D printer ready
+
+ unsigned char AxisPagenum = 0; // 0 for 10mm, 1 for 1mm, 2 for 0.1mm
+ bool InforShowStatus = true;
+ bool TPShowStatus = false; // true for only opening time and percentage, false for closing time and percentage.
+ bool AutohomeKey = false;
+ unsigned char AutoHomeIconNum;
+ int16_t userConfValidation = 0;
+
+ uint8_t lastPauseMsgState = 0;
+
+ creality_dwin_settings_t Settings;
+ uint8_t dwin_settings_version = 1;
+
+ bool reEntryPrevent = false;
+ uint8_t reEntryCount = 0;
+ uint16_t idleThrottling = 0;
+
+ bool pause_resume_selected = false;
+
+ #if HAS_PID_HEATING
+ uint16_t pid_hotendAutoTemp = 150;
+ uint16_t pid_bedAutoTemp = 70;
+ #endif
+
+ void onStartup() {
+ DWIN_SERIAL.begin(115200);
+ rtscheck.recdat.head[0] = rtscheck.snddat.head[0] = FHONE;
+ rtscheck.recdat.head[1] = rtscheck.snddat.head[1] = FHTWO;
+ memset(rtscheck.databuf, 0, sizeof(rtscheck.databuf));
+
+ #if ENABLED(DWINOS_4)
+ #define DWIN_BOOTUP_DELAY 1500
+ #else
+ #define DWIN_BOOTUP_DELAY 500
+ #endif
+ delay_ms(DWIN_BOOTUP_DELAY); // Delay to allow screen startup
+ SetTouchScreenConfiguration();
+ rtscheck.RTS_SndData(StartSoundSet, SoundAddr);
+ delay_ms(400); // Delay to allow screen to configure
+
+ #if defined(CUSTOM_MACHINE_NAME)
+ onStatusChanged(CUSTOM_MACHINE_NAME " Ready");
+ #else
+ onStatusChanged(MACHINE_NAME " Ready");
+ #endif
+
+ rtscheck.RTS_SndData(100, FeedrateDisplay);
+
+ /***************turn off motor*****************/
+ rtscheck.RTS_SndData(11, FilenameIcon);
+
+ /***************transmit temperature to screen*****************/
+ rtscheck.RTS_SndData(0, NozzlePreheat);
+ rtscheck.RTS_SndData(0, BedPreheat);
+ rtscheck.RTS_SndData(getActualTemp_celsius(H0), NozzleTemp);
+ #if HAS_MULTI_HOTEND
+ rtscheck.RTS_SndData(getActualTemp_celsius(H1), e2Temp);
+ #else
+ rtscheck.RTS_SndData(0, e2Temp);
+ #endif
+ rtscheck.RTS_SndData(getActualTemp_celsius(BED), Bedtemp);
+ /***************transmit Fan speed to screen*****************/
+ rtscheck.RTS_SndData(getActualFan_percent((fan_t)getActiveTool()), FanKeyIcon);
+
+ /***************transmit Printer information to screen*****************/
+ for (int j = 0; j < 20; j++) // clean filename
+ rtscheck.RTS_SndData(0, MacVersion + j);
+ char sizebuf[20] = {0};
+ sprintf(sizebuf, "%d X %d X %d", Y_BED_SIZE, X_BED_SIZE, Z_MAX_POS);
+ #if defined(CUSTOM_MACHINE_NAME)
+ rtscheck.RTS_SndData(CUSTOM_MACHINE_NAME, MacVersion);
+ #else
+ rtscheck.RTS_SndData(MACHINE_NAME, MacVersion);
+ #endif
+ rtscheck.RTS_SndData(SHORT_BUILD_VERSION, SoftVersion);
+ rtscheck.RTS_SndData(sizebuf, PrinterSize);
+ rtscheck.RTS_SndData(WEBSITE_URL, CorpWebsite);
+
+ /**************************some info init*******************************/
+ rtscheck.RTS_SndData(0, PrintscheduleIcon);
+ rtscheck.RTS_SndData(0, PrintscheduleIcon + 1);
+
+ /************************clean screen*******************************/
+ for (int i = 0; i < MaxFileNumber; i++)
+ for (int j = 0; j < 10; j++) rtscheck.RTS_SndData(0, SDFILE_ADDR + i * 10 + j);
+
+ for (int j = 0; j < 10; j++) {
+ rtscheck.RTS_SndData(0, Printfilename + j); // clean screen.
+ rtscheck.RTS_SndData(0, Choosefilename + j); // clean filename
+ }
+ for (int j = 0; j < 8; j++) rtscheck.RTS_SndData(0, FilenameCount + j);
+ for (int j = 1; j <= MaxFileNumber; j++) {
+ rtscheck.RTS_SndData(10, FilenameIcon + j);
+ rtscheck.RTS_SndData(10, FilenameIcon1 + j);
+ }
+
+ DEBUG_ECHOLNPGM("==Dwin Init Complete==");
+ }
+
+ void onIdle() {
+
+ while (rtscheck.RTS_RecData() > 0 && (rtscheck.recdat.data[0] != 0 || rtscheck.recdat.addr != 0)) rtscheck.RTS_HandleData();
+
+ if (reEntryPrevent && reEntryCount < 120) {
+ reEntryCount++;
+ return;
+ }
+ reEntryCount = 0;
+
+ if (idleThrottling++ < 750)
+ return;
+
+ // Always send temperature data
+ rtscheck.RTS_SndData(getActualTemp_celsius(getActiveTool()), NozzleTemp);
+ rtscheck.RTS_SndData(getActualTemp_celsius(BED), Bedtemp);
+ rtscheck.RTS_SndData(getTargetTemp_celsius(getActiveTool()), NozzlePreheat);
+ rtscheck.RTS_SndData(getTargetTemp_celsius(BED), BedPreheat);
+ #if HAS_MULTI_HOTEND
+ rtscheck.RTS_SndData(getActualTemp_celsius(H1), e2Temp);
+ rtscheck.RTS_SndData(getTargetTemp_celsius(H1), e2Preheat);
+
+ rtscheck.RTS_SndData(((uint8_t)getActiveTool() + 1), ActiveToolVP);
+ #else
+ rtscheck.RTS_SndData(0, e2Temp);
+ rtscheck.RTS_SndData(0, e2Preheat);
+ #endif
+
+ if (awaitingUserConfirm() && (lastPauseMsgState != ExtUI::pauseModeStatus || userConfValidation > 99)) {
+ switch (ExtUI::pauseModeStatus) {
+ case PAUSE_MESSAGE_PARKING: ExtUI::onUserConfirmRequired(GET_TEXT_F(MSG_PAUSE_PRINT_PARKING)); break;
+ case PAUSE_MESSAGE_CHANGING: ExtUI::onUserConfirmRequired(GET_TEXT_F(MSG_FILAMENT_CHANGE_INIT)); break;
+ case PAUSE_MESSAGE_UNLOAD: ExtUI::onUserConfirmRequired(GET_TEXT_F(MSG_FILAMENT_CHANGE_UNLOAD)); break;
+ case PAUSE_MESSAGE_WAITING: ExtUI::onUserConfirmRequired(GET_TEXT_F(MSG_ADVANCED_PAUSE_WAITING)); break;
+ case PAUSE_MESSAGE_INSERT: ExtUI::onUserConfirmRequired(GET_TEXT_F(MSG_FILAMENT_CHANGE_INSERT)); break;
+ case PAUSE_MESSAGE_LOAD: ExtUI::onUserConfirmRequired(GET_TEXT_F(MSG_FILAMENT_CHANGE_LOAD)); break;
+ case PAUSE_MESSAGE_PURGE:
+ #if ENABLED(ADVANCED_PAUSE_CONTINUOUS_PURGE)
+ ExtUI::onUserConfirmRequired(GET_TEXT_F(MSG_FILAMENT_CHANGE_CONT_PURGE)); break;
+ #else
+ ExtUI::onUserConfirmRequired(GET_TEXT_F(MSG_FILAMENT_CHANGE_PURGE)); break;
+ #endif
+ case PAUSE_MESSAGE_RESUME: ExtUI::onUserConfirmRequired(GET_TEXT_F(MSG_FILAMENT_CHANGE_RESUME)); break;
+ case PAUSE_MESSAGE_HEAT: ExtUI::onUserConfirmRequired(GET_TEXT_F(MSG_FILAMENT_CHANGE_HEAT)); break;
+ case PAUSE_MESSAGE_HEATING: ExtUI::onUserConfirmRequired(GET_TEXT_F(MSG_FILAMENT_CHANGE_HEATING)); break;
+ case PAUSE_MESSAGE_OPTION: ExtUI::onUserConfirmRequired(GET_TEXT_F(MSG_FILAMENT_CHANGE_OPTION_HEADER)); break;
+ case PAUSE_MESSAGE_STATUS: DEBUG_ECHOLNPGM("PauseStatus"); break;
+ default: onUserConfirmRequired(PSTR("Confirm Continue")); break;
+ }
+ userConfValidation = 0;
+
+ }
+ else if (pause_resume_selected && !awaitingUserConfirm()) {
+ rtscheck.RTS_SndData(ExchangePageBase + 53, ExchangepageAddr);
+ pause_resume_selected = false;
+ userConfValidation = 0;
+ }
+ else if (awaitingUserConfirm() && !pause_resume_selected) {
+ userConfValidation++;
+ }
+ else if (awaitingUserConfirm() && pause_resume_selected) {
+ pause_resume_selected = false;
+ userConfValidation = 100;
+ }
+
+ reEntryPrevent = true;
+ idleThrottling = 0;
+ if (waitway && !commandsInQueue())
+ waitway_lock++;
+ else
+ waitway_lock = 0;
+
+ if (waitway_lock > 100) {
+ waitway_lock = 0;
+ waitway = 0; // clear waitway if nothing is going on
+ }
+
+ switch (waitway) {
+ case 1:
+ if (isPositionKnown()) {
+ InforShowStatus = true;
+ DEBUG_ECHOLNPGM("==waitway 1==");
+ rtscheck.RTS_SndData(ExchangePageBase + 54, ExchangepageAddr);
+ waitway = 0;
+ }
+ break;
+
+ case 2:
+ if (isPositionKnown() && !commandsInQueue())
+ waitway = 0;
+ break;
+
+ case 3:
+ DEBUG_ECHOLNPGM("==waitway 3==");
+ // if(isPositionKnown() && (getActualTemp_celsius(BED) >= (getTargetTemp_celsius(BED)-1))) {
+ rtscheck.RTS_SndData(ExchangePageBase + 64, ExchangepageAddr);
+ waitway = 7;
+ // return;
+ // }
+ break;
+
+ case 4:
+ if (AutohomeKey && isPositionKnown() && !commandsInQueue()) { // Manual Move Home Done
+ DEBUG_ECHOLNPGM("==waitway 4==");
+ // rtscheck.RTS_SndData(ExchangePageBase + 71 + AxisPagenum, ExchangepageAddr);
+ AutohomeKey = false;
+ waitway = 0;
+ }
+ break;
+ case 5:
+ if (isPositionKnown() && !commandsInQueue()) {
+ InforShowStatus = true;
+ waitway = 0;
+ DEBUG_ECHOLNPGM("==waitway 5==");
+ rtscheck.RTS_SndData(ExchangePageBase + 78, ExchangepageAddr); // exchange to 78 page
+ }
+ break;
+ case 6:
+ if (!commandsInQueue()) {
+ setAxisPosition_mm(BED_TRAMMING_HEIGHT, (axis_t)Z);
+ waitway = 0;
+ }
+ break;
+ case 7:
+ if (!commandsInQueue())
+ waitway = 0;
+ break;
+ }
+
+ void yield();
+
+ #if HAS_MESH
+ if (getLevelingActive())
+ rtscheck.RTS_SndData(3, AutoLevelIcon); /*On*/
+ else
+ rtscheck.RTS_SndData(2, AutoLevelIcon); /*Off*/
+ #endif
+
+ #if HAS_FILAMENT_SENSOR
+ if (getFilamentRunoutEnabled())
+ rtscheck.RTS_SndData(3, RunoutToggle); /*On*/
+ else
+ rtscheck.RTS_SndData(2, RunoutToggle); /*Off*/
+ #endif
+
+ #if ENABLED(CASE_LIGHT_ENABLE)
+ if (getCaseLightState())
+ rtscheck.RTS_SndData(3, LedToggle); /*On*/
+ else
+ rtscheck.RTS_SndData(2, LedToggle); /*Off*/
+ #endif
+
+ #if ENABLED(POWER_LOSS_RECOVERY)
+ if (getPowerLossRecoveryEnabled())
+ rtscheck.RTS_SndData(3, PowerLossToggle); /*On*/
+ else
+ rtscheck.RTS_SndData(2, PowerLossToggle); /*Off*/
+ #endif
+
+ if (startprogress == 0) {
+ startprogress += 25;
+ delay_ms(3000); // Delay to show bootscreen
+ }
+ else if (startprogress < 250) {
+ if (isMediaInserted()) // Re init media as it happens too early on STM32 boards often
+ onMediaInserted();
+ else
+ injectCommands(F("M22\nM21"));
+ startprogress = 254;
+ DEBUG_ECHOLNPGM(" startprogress ");
+ InforShowStatus = true;
+ TPShowStatus = false;
+ rtscheck.RTS_SndData(ExchangePageBase + 45, ExchangepageAddr);
+ reEntryPrevent = false;
+ return;
+ }
+ if (startprogress <= 100)
+ rtscheck.RTS_SndData(startprogress, StartIcon);
+ else
+ rtscheck.RTS_SndData((startprogress - 100), StartIcon + 1);
+
+ // rtscheck.RTS_SndData((startprogress++) % 5, ExchFlmntIcon);
+
+ if (isPrinting()) {
+ rtscheck.RTS_SndData(getActualFan_percent((fan_t)getActiveTool()), FanKeyIcon);
+ rtscheck.RTS_SndData(getProgress_seconds_elapsed() / 3600, Timehour);
+ rtscheck.RTS_SndData((getProgress_seconds_elapsed() % 3600) / 60, Timemin);
+ if (getProgress_percent() > 0) {
+ Percentrecord = getProgress_percent() + 1;
+ if (Percentrecord <= 50) {
+ rtscheck.RTS_SndData((unsigned int)Percentrecord * 2, PrintscheduleIcon);
+ rtscheck.RTS_SndData(0, PrintscheduleIcon + 1);
+ }
+ else {
+ rtscheck.RTS_SndData(100, PrintscheduleIcon);
+ rtscheck.RTS_SndData((unsigned int)Percentrecord * 2 - 100, PrintscheduleIcon + 1);
+ }
+ }
+ else {
+ rtscheck.RTS_SndData(0, PrintscheduleIcon);
+ rtscheck.RTS_SndData(0, PrintscheduleIcon + 1);
+ }
+ rtscheck.RTS_SndData((unsigned int)getProgress_percent(), Percentage);
+ }
+ else { // Not printing settings
+ rtscheck.RTS_SndData(map(constrain(Settings.display_volume, 0, 255), 0, 255, 0, 100), VolumeDisplay);
+ rtscheck.RTS_SndData(Settings.screen_brightness, DisplayBrightness);
+ rtscheck.RTS_SndData(Settings.standby_screen_brightness, DisplayStandbyBrightness);
+ rtscheck.RTS_SndData(Settings.standby_time_seconds, DisplayStandbySeconds);
+ if (Settings.display_standby)
+ rtscheck.RTS_SndData(3, DisplayStandbyEnableIndicator);
+ else
+ rtscheck.RTS_SndData(2, DisplayStandbyEnableIndicator);
+
+ rtscheck.RTS_SndData((unsigned int)(getAxisSteps_per_mm(X) * 10), StepMM_X);
+ rtscheck.RTS_SndData((unsigned int)(getAxisSteps_per_mm(Y) * 10), StepMM_Y);
+ rtscheck.RTS_SndData((unsigned int)(getAxisSteps_per_mm(Z) * 10), StepMM_Z);
+ rtscheck.RTS_SndData((unsigned int)(getAxisSteps_per_mm(E0) * 10), StepMM_E);
+
+ rtscheck.RTS_SndData(((unsigned int)getAxisMaxAcceleration_mm_s2(X) / 100), Accel_X);
+ rtscheck.RTS_SndData(((unsigned int)getAxisMaxAcceleration_mm_s2(Y) / 100), Accel_Y);
+ rtscheck.RTS_SndData(((unsigned int)getAxisMaxAcceleration_mm_s2(Z) / 10), Accel_Z);
+ rtscheck.RTS_SndData(((unsigned int)getAxisMaxAcceleration_mm_s2(E0)), Accel_E);
+
+ rtscheck.RTS_SndData(((unsigned int)getAxisMaxFeedrate_mm_s(X)), Feed_X);
+ rtscheck.RTS_SndData(((unsigned int)getAxisMaxFeedrate_mm_s(Y)), Feed_Y);
+ rtscheck.RTS_SndData(((unsigned int)getAxisMaxFeedrate_mm_s(Z)), Feed_Z);
+ rtscheck.RTS_SndData(((unsigned int)getAxisMaxFeedrate_mm_s(E0)), Feed_E);
+
+ rtscheck.RTS_SndData(((unsigned int)getAxisMaxJerk_mm_s(X) * 100), Jerk_X);
+ rtscheck.RTS_SndData(((unsigned int)getAxisMaxJerk_mm_s(Y) * 100), Jerk_Y);
+ rtscheck.RTS_SndData(((unsigned int)getAxisMaxJerk_mm_s(Z) * 100), Jerk_Z);
+ rtscheck.RTS_SndData(((unsigned int)getAxisMaxJerk_mm_s(E0) * 100), Jerk_E);
+
+ #if HAS_HOTEND_OFFSET
+ rtscheck.RTS_SndData(((unsigned int)getNozzleOffset_mm(X, E1) * 10), T2Offset_X);
+ rtscheck.RTS_SndData(((unsigned int)getNozzleOffset_mm(Y, E1) * 10), T2Offset_Y);
+ rtscheck.RTS_SndData(((unsigned int)getNozzleOffset_mm(Z, E1) * 10), T2Offset_Z);
+ rtscheck.RTS_SndData((unsigned int)(getAxisSteps_per_mm(E1) * 10), T2StepMM_E);
+ #endif
+
+ #if HAS_BED_PROBE
+ rtscheck.RTS_SndData(getProbeOffset_mm(X) * 100, ProbeOffset_X);
+ rtscheck.RTS_SndData(getProbeOffset_mm(Y) * 100, ProbeOffset_Y);
+ #endif
+
+ #if HAS_PID_HEATING
+ rtscheck.RTS_SndData(pid_hotendAutoTemp, HotendPID_AutoTmp);
+ rtscheck.RTS_SndData(pid_bedAutoTemp, BedPID_AutoTmp);
+ rtscheck.RTS_SndData((unsigned int)(getPID_Kp(E0) * 10), HotendPID_P);
+ rtscheck.RTS_SndData((unsigned int)(getPID_Ki(E0) * 10), HotendPID_I);
+ rtscheck.RTS_SndData((unsigned int)(getPID_Kd(E0) * 10), HotendPID_D);
+ #if ENABLED(PIDTEMPBED)
+ rtscheck.RTS_SndData((unsigned int)(getBedPID_Kp() * 10), BedPID_P);
+ rtscheck.RTS_SndData((unsigned int)(getBedPID_Ki() * 10), BedPID_I);
+ rtscheck.RTS_SndData((unsigned int)(getBedPID_Kd() * 10), BedPID_D);
+ #endif
+ #endif
+ }
+
+ rtscheck.RTS_SndData(getZOffset_mm() * 100, ProbeOffset_Z);
+ rtscheck.RTS_SndData((unsigned int)(getFlow_percent(E0)), Flowrate);
+
+ if (NozzleTempStatus[0] || NozzleTempStatus[2]) { // statuse of loadfilement and unloadfinement when temperature is less than
+ unsigned int IconTemp;
+
+ IconTemp = getActualTemp_celsius(getActiveTool()) * 100 / getTargetTemp_celsius(getActiveTool());
+ if (IconTemp >= 100)
+ IconTemp = 100;
+ rtscheck.RTS_SndData(IconTemp, HeatPercentIcon);
+ if (getActualTemp_celsius(getActiveTool()) > EXTRUDE_MINTEMP && NozzleTempStatus[0] != 0) {
+ NozzleTempStatus[0] = 0;
+ rtscheck.RTS_SndData(10 * ChangeMaterialbuf[0], FilamentUnit1);
+ rtscheck.RTS_SndData(10 * ChangeMaterialbuf[1], FilamentUnit2);
+ DEBUG_ECHOLNPGM("==Heating Done Change Filament==");
+ rtscheck.RTS_SndData(ExchangePageBase + 65, ExchangepageAddr);
+ }
+ else if (getActualTemp_celsius(getActiveTool()) >= getTargetTemp_celsius(getActiveTool()) && NozzleTempStatus[2]) {
+ DEBUG_ECHOLNPGM("***NozzleTempStatus[2] =", (int)NozzleTempStatus[2]);
+ NozzleTempStatus[2] = 0;
+ TPShowStatus = true;
+ rtscheck.RTS_SndData(4, ExchFlmntIcon);
+ rtscheck.RTS_SndData(ExchangePageBase + 83, ExchangepageAddr);
+ }
+ else if (NozzleTempStatus[2]) {
+ // rtscheck.RTS_SndData((startprogress++) % 5, ExchFlmntIcon);
+ }
+ }
+
+ if (AutohomeKey) {
+ rtscheck.RTS_SndData(AutoHomeIconNum, AutoZeroIcon);
+ if (++AutoHomeIconNum > 9) AutoHomeIconNum = 0;
+ }
+
+ if (isMediaInserted()) {
+ uint16_t currPage, maxPageAdd;
+ if (fileIndex == 0)
+ currPage = 1;
+ else
+ currPage = CEIL((float)((float)fileIndex / (float)DISPLAY_FILES)) + 1;
+
+ if (filenavigator.folderdepth != 0)
+ maxPageAdd = 1;
+ else
+ maxPageAdd = 0;
+ uint16_t maxPages = CEIL((float)(filenavigator.maxFiles() + maxPageAdd) / (float)DISPLAY_FILES );
+
+ rtscheck.RTS_SndData(currPage, FilesCurentPage);
+ rtscheck.RTS_SndData(maxPages, FilesMaxPage);
+ }
+ else {
+ rtscheck.RTS_SndData(0, FilesCurentPage);
+ rtscheck.RTS_SndData(0, FilesMaxPage);
+ }
+
+ void yield();
+
+ if (rtscheck.recdat.addr != DisplayZaxis && rtscheck.recdat.addr != DisplayYaxis && rtscheck.recdat.addr != DisplayZaxis) {
+ rtscheck.RTS_SndData(10 * getAxisPosition_mm((axis_t)X), DisplayXaxis);
+ rtscheck.RTS_SndData(10 * getAxisPosition_mm((axis_t)Y), DisplayYaxis);
+ rtscheck.RTS_SndData(10 * getAxisPosition_mm((axis_t)Z), DisplayZaxis);
+ }
+ reEntryPrevent = false;
+ }
+
+
+ rx_datagram_state_t RTSSHOW::rx_datagram_state = DGUS_IDLE;
+ uint8_t RTSSHOW::rx_datagram_len = 0;
+ bool RTSSHOW::Initialized = false;
+ RTSSHOW::RTSSHOW() {
+ recdat.head[0] = snddat.head[0] = FHONE;
+ recdat.head[1] = snddat.head[1] = FHTWO;
+ memset(databuf, 0, sizeof(databuf));
+ }
+
+ int RTSSHOW::RTS_RecData() {
+ uint8_t receivedbyte;
+ while (DWIN_SERIAL.available())
+ switch (rx_datagram_state) {
+
+ case DGUS_IDLE: // Waiting for the first header byte
+ receivedbyte = DWIN_SERIAL.read();
+ //DEBUG_ECHOLNPGM("< ",receivedbyte);
+ if (FHONE == receivedbyte) rx_datagram_state = DGUS_HEADER1_SEEN;
+ break;
+
+ case DGUS_HEADER1_SEEN: // Waiting for the second header byte
+ receivedbyte = DWIN_SERIAL.read();
+ //DEBUG_ECHOLNPGM(" ", receivedbyte);
+ rx_datagram_state = (FHTWO == receivedbyte) ? DGUS_HEADER2_SEEN : DGUS_IDLE;
+ break;
+
+ case DGUS_HEADER2_SEEN: // Waiting for the length byte
+ rx_datagram_len = DWIN_SERIAL.read();
+ //DEBUGLCDCOMM_ECHOPAIR(" (", rx_datagram_len, ") ");
+
+ // Telegram min len is 3 (command and one word of payload)
+ rx_datagram_state = WITHIN(rx_datagram_len, 3, DGUS_RX_BUFFER_SIZE) ? DGUS_WAIT_TELEGRAM : DGUS_IDLE;
+ break;
+
+ case DGUS_WAIT_TELEGRAM: // wait for complete datagram to arrive.
+ if (DWIN_SERIAL.available() < rx_datagram_len) return -1;
+
+ Initialized = true; // We've talked to it, so we defined it as initialized.
+ uint8_t command = DWIN_SERIAL.read();
+
+ // DEBUGLCDCOMM_ECHOPAIR("# ", command);
+
+ uint8_t readlen = rx_datagram_len - 1; // command is part of len.
+ unsigned char tmp[rx_datagram_len - 1];
+ unsigned char *ptmp = tmp;
+ while (readlen--) {
+ receivedbyte = DWIN_SERIAL.read();
+ // DEBUGLCDCOMM_ECHOPAIR(" ", receivedbyte);
+ *ptmp++ = receivedbyte;
+ }
+ // DEBUGLCDCOMM_ECHOPGM(" # ");
+ // mostly we'll get this: 5A A5 03 82 4F 4B -- ACK on 0x82, so discard it.
+ if (command == VarAddr_W && 'O' == tmp[0] && 'K' == tmp[1]) {
+ // DEBUG_ECHOLNPGM(">");
+ rx_datagram_state = DGUS_IDLE;
+ break;
+ }
+
+ /* AutoUpload, (and answer to) Command 0x83 :
+ | tmp[0 1 2 3 4 ... ]
+ | Example 5A A5 06 83 20 01 01 78 01 ……
+ | / / | | \ / | \ \
+ | Header | | | | \_____\_ DATA (Words!)
+ | DatagramLen / VPAdr |
+ | Command DataLen (in Words) */
+ if (command == VarAddr_R) {
+ const uint16_t vp = tmp[0] << 8 | tmp[1];
+
+ const uint8_t dlen = tmp[2] << 1; // Convert to Bytes. (Display works with words)
+ //DEBUG_ECHOLNPGM(" vp=", vp, " dlen=", dlen);
+ recdat.addr = vp;
+ recdat.len = tmp[2];
+ for (unsigned int i = 0; i < dlen; i += 2) {
+ recdat.data[i / 2] = tmp[3 + i];
+ recdat.data[i / 2] = (recdat.data[i / 2] << 8 ) | tmp[4 + i];
+ }
+
+ DEBUG_ECHOLNPGM("VP received: ", vp, " - len ", tmp[2]);
+
+ DEBUG_ECHOLNPGM("d1: ", tmp[3], " - d2 ", tmp[4]);
+ DEBUG_ECHOLNPGM("d3: ", tmp[5], " - d4 ", tmp[6]);
+
+ rx_datagram_state = DGUS_IDLE;
+ return 2;
+ break;
+ }
+
+ // discard anything else
+ rx_datagram_state = DGUS_IDLE;
+ }
+ return -1;
+ }
+
+ void RTSSHOW::RTS_SndData(void) {
+ if ((snddat.head[0] == FHONE) && (snddat.head[1] == FHTWO) && snddat.len >= 3) {
+ databuf[0] = snddat.head[0];
+ databuf[1] = snddat.head[1];
+ databuf[2] = snddat.len;
+ databuf[3] = snddat.command;
+ if (snddat.command == 0x80) { // to write data to the register
+ databuf[4] = snddat.addr;
+ for (int i = 0; i < (snddat.len - 2); i++) databuf[5 + i] = snddat.data[i];
+ }
+ else if (snddat.len == 3 && (snddat.command == 0x81)) { // to read data from the register
+ databuf[4] = snddat.addr;
+ databuf[5] = snddat.bytelen;
+ }
+ else if (snddat.command == 0x82) { // to write data to the variate
+ databuf[4] = snddat.addr >> 8;
+ databuf[5] = snddat.addr & 0xFF;
+ for (int i = 0; i < (snddat.len - 3); i += 2) {
+ databuf[6 + i] = snddat.data[i / 2] >> 8;
+ databuf[7 + i] = snddat.data[i / 2] & 0xFF;
+ }
+ }
+ else if (snddat.len == 4 && (snddat.command == 0x83)) { // to read data from the variate
+ databuf[4] = snddat.addr >> 8;
+ databuf[5] = snddat.addr & 0xFF;
+ databuf[6] = snddat.bytelen;
+ }
+ for (int i = 0; i < (snddat.len + 3); i++) {
+ DWIN_SERIAL.write(databuf[i]);
+ delay_us(1);
+ }
+
+ memset(&snddat, 0, sizeof(snddat));
+ memset(databuf, 0, sizeof(databuf));
+ snddat.head[0] = FHONE;
+ snddat.head[1] = FHTWO;
+ }
+ }
+
+ void RTSSHOW::RTS_SndData(const String &s, unsigned long addr, unsigned char cmd /*= VarAddr_W*/) {
+ if (s.length() < 1)
+ return;
+ RTS_SndData(s.c_str(), addr, cmd);
+ }
+
+ void RTSSHOW::RTS_SndData(const char *str, unsigned long addr, unsigned char cmd /*= VarAddr_W*/) {
+
+ int len = strlen(str);
+ constexpr int maxlen = SizeofDatabuf - 6;
+ if (len > 0) {
+ if (len > maxlen) len = maxlen;
+ databuf[0] = FHONE;
+ databuf[1] = FHTWO;
+ databuf[2] = 3 + len;
+ databuf[3] = cmd;
+ databuf[4] = addr >> 8;
+ databuf[5] = addr & 0x00FF;
+ for (int i = 0; i < len; i++) databuf[6 + i] = str[i];
+
+ for (int i = 0; i < (len + 6); i++) {
+ DWIN_SERIAL.write(databuf[i]);
+ delay_us(1);
+ }
+ memset(databuf, 0, sizeof(databuf));
+ }
+ }
+
+ void RTSSHOW::RTS_SndData(char c, unsigned long addr, unsigned char cmd /*= VarAddr_W*/) {
+ snddat.command = cmd;
+ snddat.addr = addr;
+ snddat.data[0] = (unsigned long)c;
+ snddat.data[0] = snddat.data[0] << 8;
+ snddat.len = 5;
+ RTS_SndData();
+ }
+
+ void RTSSHOW::RTS_SndData(unsigned char *str, unsigned long addr, unsigned char cmd) { RTS_SndData((char *)str, addr, cmd); }
+
+ void RTSSHOW::RTS_SndData(int n, unsigned long addr, unsigned char cmd /*= VarAddr_W*/) {
+ if (cmd == VarAddr_W) {
+ if ((uint8_t)n > 0xFFFF) {
+ snddat.data[0] = n >> 16;
+ snddat.data[1] = n & 0xFFFF;
+ snddat.len = 7;
+ }
+ else {
+ snddat.data[0] = n;
+ snddat.len = 5;
+ }
+ }
+ else if (cmd == RegAddr_W) {
+ snddat.data[0] = n;
+ snddat.len = 3;
+ }
+ else if (cmd == VarAddr_R) {
+ snddat.bytelen = n;
+ snddat.len = 4;
+ }
+ snddat.command = cmd;
+ snddat.addr = addr;
+ RTS_SndData();
+ }
+
+ void RTSSHOW::RTS_SndData(unsigned int n, unsigned long addr, unsigned char cmd) { RTS_SndData((int)n, addr, cmd); }
+
+ void RTSSHOW::RTS_SndData(float n, unsigned long addr, unsigned char cmd) { RTS_SndData((int)n, addr, cmd); }
+
+ void RTSSHOW::RTS_SndData(long n, unsigned long addr, unsigned char cmd) { RTS_SndData((unsigned long)n, addr, cmd); }
+
+ void RTSSHOW::RTS_SndData(unsigned long n, unsigned long addr, unsigned char cmd /*= VarAddr_W*/) {
+ if (cmd == VarAddr_W) {
+ if (n > 0xFFFF) {
+ snddat.data[0] = n >> 16;
+ snddat.data[1] = n & 0xFFFF;
+
+ // snddat.data[0] = n >> 24;
+ // snddat.data[1] = n >> 16;
+ // snddat.data[2] = n >> 8;
+ // snddat.data[3] = n;
+ snddat.len = 7;
+ }
+ else {
+ snddat.data[0] = n;
+ snddat.len = 5;
+ }
+ }
+ else if (cmd == VarAddr_R) {
+ snddat.bytelen = n;
+ snddat.len = 4;
+ }
+ snddat.command = cmd;
+ snddat.addr = addr;
+ RTS_SndData();
+ }
+
+ void RTSSHOW::RTS_HandleData() {
+ int Checkkey = -1;
+ DEBUG_ECHOLNPGM(" *******RTS_HandleData******** ");
+ if (waitway > 0) { // for waiting
+ DEBUG_ECHOLNPGM("waitway ==", (int)waitway);
+ memset(&recdat, 0, sizeof(recdat));
+ recdat.head[0] = FHONE;
+ recdat.head[1] = FHTWO;
+ return;
+ }
+ DEBUG_ECHOLNPGM("recdat.data[0] ==", recdat.data[0]);
+ DEBUG_ECHOLNPGM("recdat.addr ==", recdat.addr);
+ for (int i = 0; Addrbuf[i] != 0; i++)
+ if (recdat.addr == Addrbuf[i]) {
+ if (Addrbuf[i] == NzBdSet || Addrbuf[i] == NozzlePreheat || Addrbuf[i] == BedPreheat || Addrbuf[i] == Flowrate)
+ Checkkey = ManualSetTemp;
+ else if (Addrbuf[i] >= Stopprint && Addrbuf[i] <= Resumeprint)
+ Checkkey = PrintChoice;
+ else if (Addrbuf[i] >= AutoZero && Addrbuf[i] <= DisplayZaxis)
+ Checkkey = XYZEaxis;
+ else if (Addrbuf[i] >= FilamentUnit1 && Addrbuf[i] <= FilamentUnit2)
+ Checkkey = Filament;
+ else
+ Checkkey = i;
+ break;
+ }
+
+ switch (recdat.addr) {
+ case ProbeOffset_Z: {
+ Checkkey = Zoffset_Value;
+ break;
+ }
+ case Flowrate:
+ case StepMM_X:
+ case StepMM_Y:
+ case StepMM_Z:
+ case StepMM_E:
+ case ProbeOffset_X:
+ case ProbeOffset_Y:
+ case HotendPID_AutoTmp:
+ case BedPID_AutoTmp:
+ case HotendPID_P:
+ case HotendPID_I:
+ case HotendPID_D:
+ case BedPID_P:
+ case BedPID_I:
+ case BedPID_D:
+ case T2Offset_X:
+ case T2Offset_Y:
+ case T2Offset_Z:
+ case T2StepMM_E:
+ case Accel_X:
+ case Accel_Y:
+ case Accel_Z:
+ case Accel_E:
+ case Feed_X:
+ case Feed_Y:
+ case Feed_Z:
+ case Feed_E:
+ case Jerk_X:
+ case Jerk_Y:
+ case Jerk_Z:
+ case Jerk_E:
+ case RunoutToggle:
+ case PowerLossToggle:
+ case FanKeyIcon:
+ case LedToggle:
+ case e2Preheat:
+ Checkkey = ManualSetTemp;
+ break;
+ }
+
+ if (recdat.addr == VolumeDisplay)
+ Checkkey = VolumeDisplay;
+ if (recdat.addr == DisplayBrightness)
+ Checkkey = DisplayBrightness;
+ if (recdat.addr == DisplayStandbyBrightness)
+ Checkkey = DisplayStandbyBrightness;
+ if (recdat.addr == DisplayStandbySeconds)
+ Checkkey = DisplayStandbySeconds;
+ if (recdat.addr >= AutolevelVal && recdat.addr <= 4400)// ((int)AutolevelVal+(GRID_MAX_POINTS_X*GRID_MAX_POINTS_Y*2)) = 4400 with 5x5 mesh
+ Checkkey = AutolevelVal;
+
+ if (recdat.addr >= SDFILE_ADDR && recdat.addr <= (SDFILE_ADDR + 10 * (FileNum + 1)))
+ Checkkey = Filename;
+
+ DEBUG_ECHOLNPGM("== Checkkey==");
+ DEBUG_ECHOLN(Checkkey);
+
+ if (Checkkey < 0) {
+ memset(&recdat, 0, sizeof(recdat));
+ recdat.head[0] = FHONE;
+ recdat.head[1] = FHTWO;
+ return;
+ }
+
+ constexpr float lfrb[4] = BED_TRAMMING_INSET_LFRB;
+ DEBUG_ECHOLNPGM("BeginSwitch");
+
+ switch (Checkkey) {
+ case Printfile:
+ if (recdat.data[0] == 1) { // card
+ InforShowStatus = false;
+ DEBUG_ECHOLNPGM("Handle Data PrintFile Pre");
+ filenavigator.getFiles(0);
+ fileIndex = 0;
+ recordcount = 0;
+ DEBUG_ECHOLNPGM("Handle Data PrintFile Post");
+ RTS_SndData(ExchangePageBase + 46, ExchangepageAddr);
+ }
+ else if (recdat.data[0] == 2) { // return after printing result.
+ InforShowStatus = true;
+ TPShowStatus = false;
+ stopPrint();
+ injectCommands(F("M84"));
+ RTS_SndData(11, FilenameIcon);
+ RTS_SndData(0, PrintscheduleIcon);
+ RTS_SndData(0, PrintscheduleIcon + 1);
+ RTS_SndData(0, Percentage);
+ delay_ms(2);
+ RTS_SndData(0, Timehour);
+ RTS_SndData(0, Timemin);
+
+ DEBUG_ECHOLNPGM("Handle Data PrintFile 2 Setting Screen ");
+ RTS_SndData(ExchangePageBase + 45, ExchangepageAddr); // exchange to 45 page
+ }
+ else if (recdat.data[0] == 3) { // Temperature control
+ InforShowStatus = true;
+ TPShowStatus = false;
+ DEBUG_ECHOLNPGM("Handle Data PrintFile 3 Setting Screen ");
+ if (getTargetFan_percent((fan_t)getActiveTool()) == 0)
+ RTS_SndData(ExchangePageBase + 58, ExchangepageAddr); // exchange to 58 page, the fans off
+ else
+ RTS_SndData(ExchangePageBase + 57, ExchangepageAddr); // exchange to 57 page, the fans on
+ }
+ else if (recdat.data[0] == 4) { // Settings
+ InforShowStatus = false;
+ }
+ break;
+
+ case Adjust:
+ if (recdat.data[0] == 1) {
+ InforShowStatus = false;
+ }
+ else if (recdat.data[0] == 2) {
+ DEBUG_ECHOLNPGM("Handle Data Adjust 2 Setting Screen ");
+ InforShowStatus = true;
+ if (PrinterStatusKey[1] == 3) // during heating
+
+ RTS_SndData(ExchangePageBase + 53, ExchangepageAddr);
+ else if (PrinterStatusKey[1] == 4)
+ RTS_SndData(ExchangePageBase + 54, ExchangepageAddr);
+ else
+ RTS_SndData(ExchangePageBase + 53, ExchangepageAddr);
+ }
+ else if (recdat.data[0] == 3) {
+ if (getTargetFan_percent((fan_t)getActiveTool()) != 0) // turn on the fan
+ setTargetFan_percent(100, FAN0);
+ else // turn off the fan
+ setTargetFan_percent(0, FAN0);
+ }
+
+ break;
+
+ case Feedrate:
+ setFeedrate_percent(recdat.data[0]);
+ break;
+
+ case PrintChoice:
+ if (recdat.addr == Stopprint) {
+ DEBUG_ECHOLNPGM("StopPrint");
+ if (recdat.data[0] == 240) { // no
+ RTS_SndData(ExchangePageBase + 53, ExchangepageAddr);
+ DEBUG_ECHOLNPGM("Stop No", recdat.data[0] );
+ }
+ else {
+ RTS_SndData(ExchangePageBase + 45, ExchangepageAddr);
+ RTS_SndData(0, Timehour);
+ RTS_SndData(0, Timemin);
+ DEBUG_ECHOLNPGM("Stop Triggered", recdat.data[0] );
+ stopPrint();
+ }
+ }
+ else if (recdat.addr == Pauseprint) {
+ if (recdat.data[0] != 0xF1)
+ break;
+
+ RTS_SndData(ExchangePageBase + 54, ExchangepageAddr);
+ pausePrint();
+ }
+ else if (recdat.addr == Resumeprint && recdat.data[0] == 1) {
+ resumePrint();
+
+ PrinterStatusKey[1] = 0;
+ InforShowStatus = true;
+
+ RTS_SndData(ExchangePageBase + 53, ExchangepageAddr);
+ }
+ if (recdat.addr == Resumeprint && recdat.data[0] == 2) { // warming
+ resumePrint();
+ NozzleTempStatus[2] = 1;
+ PrinterStatusKey[1] = 0;
+ InforShowStatus = true;
+ RTS_SndData(ExchangePageBase + 82, ExchangepageAddr);
+ }
+ break;
+
+ case Zoffset:
+ float tmp_zprobe_offset;
+ if (recdat.data[0] >= 32768)
+ tmp_zprobe_offset = ((float)recdat.data[0] - 65536) / 100;
+ else
+ tmp_zprobe_offset = ((float)recdat.data[0]) / 100;
+ DEBUG_ECHOLNPGM("Requested Offset ", tmp_zprobe_offset);
+ if (WITHIN((tmp_zprobe_offset), Z_PROBE_OFFSET_RANGE_MIN, Z_PROBE_OFFSET_RANGE_MAX)) {
+ int16_t tmpSteps = mmToWholeSteps(getZOffset_mm() - tmp_zprobe_offset, (axis_t)Z);
+ if (tmpSteps == 0) {
+ DEBUG_ECHOLNPGM("Rounding to step");
+ tmpSteps = getZOffset_mm() < tmp_zprobe_offset ? 1 : -1;
+ }
+ smartAdjustAxis_steps(tmpSteps * -1, (axis_t)Z, false);
+ char zOffs[20], tmp1[11];
+ sprintf_P(zOffs, PSTR("Z Offset : %s"), dtostrf(getZOffset_mm(), 1, 3, tmp1));
+ onStatusChanged(zOffs);
+ }
+ else {
+ onStatusChanged(F("Requested Offset Beyond Limits"));
+ RTS_SndData(getZOffset_mm() * 100, ProbeOffset_Z);
+ }
+
+ rtscheck.RTS_SndData(getZOffset_mm() * 100, ProbeOffset_Z);
+ break;
+
+ case TempControl:
+ if (recdat.data[0] == 0) {
+ InforShowStatus = true;
+ TPShowStatus = false;
+ }
+ else if (recdat.data[0] == 1) {
+ if (getTargetFan_percent((fan_t)getActiveTool()) == 0)
+ RTS_SndData(ExchangePageBase + 60, ExchangepageAddr); // exchange to 60 page, the fans off
+ else
+ RTS_SndData(ExchangePageBase + 59, ExchangepageAddr); // exchange to 59 page, the fans on
+ }
+ else if (recdat.data[0] == 2) {
+ InforShowStatus = true;
+ }
+ else if (recdat.data[0] == 3) {
+ if (getTargetFan_percent((fan_t)getActiveTool()) == 0) { // turn on the fan
+ setTargetFan_percent(100, FAN0);
+ RTS_SndData(ExchangePageBase + 57, ExchangepageAddr); // exchange to 57 page, the fans on
+ }
+ else { // turn off the fan
+ setTargetFan_percent(0, FAN0);
+ RTS_SndData(ExchangePageBase + 58, ExchangepageAddr); // exchange to 58 page, the fans on
+ }
+ }
+ else if (recdat.data[0] == 5) { // PLA mode
+ setTargetTemp_celsius(PREHEAT_1_TEMP_HOTEND, getActiveTool());
+ setTargetTemp_celsius(PREHEAT_1_TEMP_BED, BED);
+
+ RTS_SndData(PREHEAT_1_TEMP_HOTEND, NozzlePreheat);
+ RTS_SndData(PREHEAT_1_TEMP_BED, BedPreheat);
+ }
+ else if (recdat.data[0] == 6) { // ABS mode
+ setTargetTemp_celsius(PREHEAT_2_TEMP_HOTEND, getActiveTool());
+ setTargetTemp_celsius(PREHEAT_2_TEMP_BED, BED);
+
+ RTS_SndData(PREHEAT_2_TEMP_HOTEND, NozzlePreheat);
+ RTS_SndData(PREHEAT_2_TEMP_BED, BedPreheat);
+ }
+ else if (recdat.data[0] == 0xF1) {
+ // InforShowStatus = true;
+ #if FAN_COUNT > 0
+ for (uint8_t i = 0; i < FAN_COUNT; i++) setTargetFan_percent(0, (fan_t)i);
+ #endif
+ setTargetTemp_celsius(0.0, H0);
+ TERN_(HAS_MULTI_HOTEND, setTargetTemp_celsius(0.0, H1));
+ setTargetTemp_celsius(0.0, BED);
+
+ RTS_SndData(0, NozzlePreheat);
+ delay_ms(1);
+ RTS_SndData(0, BedPreheat);
+ delay_ms(1);
+
+ RTS_SndData(ExchangePageBase + 57, ExchangepageAddr);
+ PrinterStatusKey[1] = 2;
+ }
+ break;
+
+ case ManualSetTemp:
+ DEBUG_ECHOLNPGM("ManualSetTemp");
+ if (recdat.addr == NzBdSet) {
+ if (recdat.data[0] == 0) {
+ if (getTargetFan_percent((fan_t)getActiveTool()) == 0)
+ RTS_SndData(ExchangePageBase + 58, ExchangepageAddr); // exchange to 58 page, the fans off
+ else
+ RTS_SndData(ExchangePageBase + 57, ExchangepageAddr); // exchange to 57 page, the fans on
+ }
+ else if (recdat.data[0] == 1) {
+ setTargetTemp_celsius(0.0, getActiveTool());
+ RTS_SndData(0, NozzlePreheat);
+ }
+ else if (recdat.data[0] == 2) {
+ setTargetTemp_celsius(0.0, BED);
+ RTS_SndData(0, BedPreheat);
+ }
+ }
+ else if (recdat.addr == NozzlePreheat) {
+ setTargetTemp_celsius((float)recdat.data[0], H0);
+ }
+ #if HAS_MULTI_HOTEND
+ else if (recdat.addr == e2Preheat) {
+ setTargetTemp_celsius((float)recdat.data[0], H1);
+ }
+ #endif
+ else if (recdat.addr == BedPreheat) {
+ setTargetTemp_celsius((float)recdat.data[0], BED);
+ }
+ else if (recdat.addr == Flowrate) {
+ setFlow_percent((int16_t)recdat.data[0], getActiveTool());
+ }
+ #if HAS_PID_HEATING
+ else if (recdat.addr == HotendPID_AutoTmp) {
+ pid_hotendAutoTemp = (uint16_t)recdat.data[0];
+ }
+ else if (recdat.addr == BedPID_AutoTmp) {
+ pid_bedAutoTemp = (uint16_t)recdat.data[0];
+ }
+ #endif
+
+ else if (recdat.addr == Accel_X) {
+ setAxisMaxAcceleration_mm_s2((uint16_t)recdat.data[0] * 100, X);
+ }
+ else if (recdat.addr == Accel_Y) {
+ setAxisMaxAcceleration_mm_s2((uint16_t)recdat.data[0] * 100, Y);
+ }
+ else if (recdat.addr == Accel_Z) {
+ setAxisMaxAcceleration_mm_s2((uint16_t)recdat.data[0] * 10, Z);
+ }
+ else if (recdat.addr == Accel_E) {
+ setAxisMaxAcceleration_mm_s2((uint16_t)recdat.data[0], E0);
+ setAxisMaxAcceleration_mm_s2((uint16_t)recdat.data[0], E1);
+ }
+
+ else if (recdat.addr == Feed_X) {
+ setAxisMaxFeedrate_mm_s((uint16_t)recdat.data[0], X);
+ }
+ else if (recdat.addr == Feed_Y) {
+ setAxisMaxFeedrate_mm_s((uint16_t)recdat.data[0], Y);
+ }
+ else if (recdat.addr == Feed_Z) {
+ setAxisMaxFeedrate_mm_s((uint16_t)recdat.data[0], Z);
+ }
+ else if (recdat.addr == Feed_E) {
+ setAxisMaxFeedrate_mm_s((uint16_t)recdat.data[0], E0);
+ setAxisMaxFeedrate_mm_s((uint16_t)recdat.data[0], E1);
+ }
+ else if (recdat.addr == FanKeyIcon) {
+ setTargetFan_percent((uint16_t)recdat.data[0], (fan_t)getActiveTool());
+ }
+ else {
+ float tmp_float_handling;
+ if (recdat.data[0] >= 32768)
+ tmp_float_handling = ((float)recdat.data[0] - 65536) / 100;
+ else
+ tmp_float_handling = ((float)recdat.data[0]) / 100;
+ if (recdat.addr == StepMM_X) {
+ setAxisSteps_per_mm(tmp_float_handling * 10, X);
+ }
+ else if (recdat.addr == StepMM_Y) {
+ setAxisSteps_per_mm(tmp_float_handling * 10, Y);
+ }
+ else if (recdat.addr == StepMM_Z) {
+ setAxisSteps_per_mm(tmp_float_handling * 10, Z);
+ }
+ else if (recdat.addr == StepMM_E) {
+ setAxisSteps_per_mm(tmp_float_handling * 10, E0);
+ #if DISABLED(DUAL_X_CARRIAGE)
+ setAxisSteps_per_mm(tmp_float_handling * 10, E1);
+ #endif
+ }
+ #if ENABLED(DUAL_X_CARRIAGE)
+ else if (recdat.addr == T2StepMM_E) {
+ setAxisSteps_per_mm(tmp_float_handling * 10, E1);
+ }
+ else if (recdat.addr == T2Offset_X) {
+ setNozzleOffset_mm(tmp_float_handling * 10, X, E1);
+ }
+ else if (recdat.addr == T2Offset_Y) {
+ setNozzleOffset_mm(tmp_float_handling * 10, Y, E1);
+ }
+ else if (recdat.addr == T2Offset_Z) {
+ setNozzleOffset_mm(tmp_float_handling * 10, Z, E1);
+ }
+ #endif
+ #if HAS_BED_PROBE
+ else if (recdat.addr == ProbeOffset_X) {
+ setProbeOffset_mm(tmp_float_handling, X);
+ }
+ else if (recdat.addr == ProbeOffset_Y) {
+ setProbeOffset_mm(tmp_float_handling, Y);
+ }
+ else if (recdat.addr == ProbeOffset_Z) {
+ setProbeOffset_mm(tmp_float_handling, Z);
+ }
+ #endif
+
+ #if ENABLED(CLASSIC_JERK)
+ else if (recdat.addr == Jerk_X) {
+ setAxisMaxJerk_mm_s(tmp_float_handling, X);
+ }
+ else if (recdat.addr == Jerk_Y) {
+ setAxisMaxJerk_mm_s(tmp_float_handling, Y);
+ }
+ else if (recdat.addr == Jerk_Z) {
+ setAxisMaxJerk_mm_s(tmp_float_handling, Z);
+ }
+ else if (recdat.addr == Jerk_E) {
+ setAxisMaxJerk_mm_s(tmp_float_handling, E0);
+ setAxisMaxJerk_mm_s(tmp_float_handling, E1);
+ }
+ #endif
+
+ #if HAS_FILAMENT_SENSOR
+ else if (recdat.addr == RunoutToggle) {
+ if (getFilamentRunoutEnabled())
+ setFilamentRunoutEnabled(false);
+ else
+ setFilamentRunoutEnabled(true);
+ }
+ #endif
+
+ #if ENABLED(POWER_LOSS_RECOVERY)
+ else if (recdat.addr == PowerLossToggle) {
+ if (getPowerLossRecoveryEnabled())
+ setPowerLossRecoveryEnabled(false);
+ else
+ setPowerLossRecoveryEnabled(true);
+ }
+ #endif
+
+ #if ENABLED(CASE_LIGHT_ENABLE)
+ else if (recdat.addr == LedToggle) {
+ if (getCaseLightState())
+ setCaseLightState(false);
+ else
+ setCaseLightState(true);
+ }
+ #endif
+
+ #if HAS_PID_HEATING
+ else if (recdat.addr == HotendPID_P) {
+ setPID(tmp_float_handling * 10, getPID_Ki(getActiveTool()), getPID_Kd(getActiveTool()), getActiveTool());
+ }
+ else if (recdat.addr == HotendPID_I) {
+ setPID(getPID_Kp(getActiveTool()), tmp_float_handling * 10, getPID_Kd(getActiveTool()), getActiveTool());
+ }
+ else if (recdat.addr == HotendPID_D) {
+ setPID(getPID_Kp(getActiveTool()), getPID_Ki(getActiveTool()), tmp_float_handling * 10, getActiveTool());
+ }
+ #if ENABLED(PIDTEMPBED)
+ else if (recdat.addr == BedPID_P) {
+ setBedPID(tmp_float_handling * 10, getBedPID_Ki(), getBedPID_Kd());
+ }
+ else if (recdat.addr == BedPID_I) {
+ setBedPID(getBedPID_Kp(), tmp_float_handling * 10, getBedPID_Kd());
+ }
+ else if (recdat.addr == BedPID_D) {
+ setBedPID(getBedPID_Kp(), getBedPID_Ki(), tmp_float_handling * 10);
+ }
+ #endif
+ #endif // HAS_PID_HEATING
+ }
+ break;
+
+ case Setting:
+ if (recdat.data[0] == 0) { // return to main page
+ InforShowStatus = true;
+ TPShowStatus = false;
+ }
+ else if (recdat.data[0] == 1) { // Bed Autoleveling
+ #if HAS_MESH
+ if (getLevelingActive())
+ RTS_SndData(3, AutoLevelIcon);
+ else
+ RTS_SndData(2, AutoLevelIcon);
+
+ if (ExtUI::getMeshValid()) {
+ uint8_t abl_probe_index = 0;
+ for (uint8_t outer = 0; outer < GRID_MAX_POINTS_Y; outer++)
+ for (uint8_t inner = 0; inner < GRID_MAX_POINTS_X; inner++) {
+ uint8_t x_Point = inner;
+ bool zig = (outer & 1);
+ if (zig) x_Point = (GRID_MAX_POINTS_X - 1) - inner;
+ xy_uint8_t point = {x_Point, outer};
+ rtscheck.RTS_SndData(ExtUI::getMeshPoint(point) * 1000, AutolevelVal + (abl_probe_index * 2));
+ ++abl_probe_index;
+ }
+ }
+ #endif
+ RTS_SndData(10, FilenameIcon); // Motor Icon
+ injectCommands(isPositionKnown() ? F("G1F1000Z0.0") : F("G28\nG1F1000Z0.0"));
+ waitway = 2;
+ RTS_SndData(ExchangePageBase + 64, ExchangepageAddr);
+ }
+ else if (recdat.data[0] == 2) { // Exchange filement
+ InforShowStatus = true;
+ TPShowStatus = false;
+ memset(ChangeMaterialbuf, 0, sizeof(ChangeMaterialbuf));
+ ChangeMaterialbuf[1] = ChangeMaterialbuf[0] = 10;
+ RTS_SndData(10 * ChangeMaterialbuf[0], FilamentUnit1); // It's ChangeMaterialbuf for show,instead of current_position[E_AXIS] in them.
+ RTS_SndData(10 * ChangeMaterialbuf[1], FilamentUnit2);
+ RTS_SndData(getActualTemp_celsius(H0), NozzleTemp);
+ RTS_SndData(getTargetTemp_celsius(H0), NozzlePreheat);
+ #if HAS_MULTI_HOTEND
+ rtscheck.RTS_SndData(getActualTemp_celsius(H1), e2Temp);
+ rtscheck.RTS_SndData(getTargetTemp_celsius(H1), e2Preheat);
+ #else
+ rtscheck.RTS_SndData(0, e2Temp);
+ rtscheck.RTS_SndData(0, e2Preheat);
+ #endif
+ delay_ms(2);
+ RTS_SndData(ExchangePageBase + 65, ExchangepageAddr);
+ }
+ else if (recdat.data[0] == 3) { // Move
+ AxisPagenum = 0;
+ RTS_SndData(ExchangePageBase + 71, ExchangepageAddr);
+ }
+ else if (recdat.data[0] == 4) { // Language
+ // Just loads language screen, now used for tools
+ }
+ else if (recdat.data[0] == 5) { // Printer Information
+ RTS_SndData(WEBSITE_URL, CorpWebsite);
+ }
+ else if (recdat.data[0] == 6) { // Diabalestepper
+ injectCommands(F("M84"));
+ RTS_SndData(11, FilenameIcon);
+ }
+ break;
+
+ case ReturnBack:
+ DEBUG_ECHOPGM("Return : ", recdat.data[0]);
+ if (recdat.data[0] == 1) { // return to the tool page
+ InforShowStatus = false;
+ RTS_SndData(ExchangePageBase + 63, ExchangepageAddr);
+ }
+ if (recdat.data[0] == 2) // return to the Level mode page
+ RTS_SndData(ExchangePageBase + 64, ExchangepageAddr);
+ break;
+
+ case Bedlevel:
+ DEBUG_ECHOLNPGM("Bed Level Option ", recdat.data[0]);
+ switch (recdat.data[0]) {
+ case 1: { // Z-axis to home
+ // Disallow Z homing if X or Y are unknown
+ if (!isAxisPositionKnown((axis_t)X) || !isAxisPositionKnown((axis_t)Y))
+ injectCommands(F("G28\nG1F1500Z0.0"));
+ else
+ injectCommands(F("G28Z\nG1F1500Z0.0"));
+
+ RTS_SndData(getZOffset_mm() * 100, ProbeOffset_Z);
+ break;
+ }
+ case 2: { // Z-axis to Up
+ if (WITHIN((getZOffset_mm() + 0.1), Z_PROBE_OFFSET_RANGE_MIN, Z_PROBE_OFFSET_RANGE_MAX)) {
+ smartAdjustAxis_steps((getAxisSteps_per_mm(Z) / 10), (axis_t)Z, false);
+ //DEBUG_ECHOLNPGM("Babystep Pos Steps : ", (int)(getAxisSteps_per_mm(Z) / 10));
+ //setZOffset_mm(getZOffset_mm() + 0.1);
+ RTS_SndData(getZOffset_mm() * 100, ProbeOffset_Z);
+ char zOffs[20], tmp1[11];
+ sprintf_P(zOffs, PSTR("Z Offset : %s"), dtostrf(getZOffset_mm(), 1, 3, tmp1));
+ onStatusChanged(zOffs);
+ }
+ break;
+ }
+ case 3: { // Z-axis to Down
+ if (WITHIN((getZOffset_mm() - 0.1), Z_PROBE_OFFSET_RANGE_MIN, Z_PROBE_OFFSET_RANGE_MAX)) {
+ smartAdjustAxis_steps(((getAxisSteps_per_mm(Z) / 10) * -1), (axis_t)Z, false);
+ // DEBUG_ECHOLNPGM("Babystep Neg Steps : ", (int)((getAxisSteps_per_mm(Z) / 10) * -1));
+ // babystepAxis_steps((((int)getAxisSteps_per_mm(Z) / 10) * -1), (axis_t)Z);
+ // setZOffset_mm(getZOffset_mm() - 0.1);
+ RTS_SndData(getZOffset_mm() * 100, ProbeOffset_Z);
+ char zOffs[20], tmp1[11];
+ sprintf_P(zOffs, PSTR("Z Offset : %s"), dtostrf(getZOffset_mm(), 1, 3, tmp1));
+ onStatusChanged(zOffs);
+ }
+ break;
+ }
+ case 4: { // Assistant Level
+ TERN_(HAS_MESH, setLevelingActive(false));
+ injectCommands(isPositionKnown() ? F("G1 F1000 Z0.0") : F("G28\nG1 F1000 Z0.0"));
+ waitway = 2;
+ RTS_SndData(ExchangePageBase + 84, ExchangepageAddr);
+ break;
+ }
+ case 5: { // AutoLevel "Measuring" Button
+ #if ENABLED(MESH_BED_LEVELING)
+ RTS_SndData(ExchangePageBase + 93, ExchangepageAddr);
+ #else
+ waitway = 3; // only for prohibiting to receive massage
+ RTS_SndData(3, AutolevelIcon);
+ uint8_t abl_probe_index = 0;
+ while (abl_probe_index < 25) {
+ rtscheck.RTS_SndData(0, AutolevelVal + abl_probe_index * 2);
+ ++abl_probe_index;
+ }
+ RTS_SndData(ExchangePageBase + 64, ExchangepageAddr);
+ injectCommands(F(MEASURING_GCODE));
+ #endif
+ break;
+ }
+
+ case 6: { // Assitant Level , Centre 1
+ setAxisPosition_mm(BED_TRAMMING_Z_HOP, (axis_t)Z);
+ setAxisPosition_mm(X_CENTER, (axis_t)X);
+ setAxisPosition_mm(Y_CENTER, (axis_t)Y);
+ waitway = 6;
+ break;
+ }
+ case 7: { // Assitant Level , Front Left 2
+ setAxisPosition_mm(BED_TRAMMING_Z_HOP, (axis_t)Z);
+ setAxisPosition_mm((X_MIN_BED + lfrb[0]), (axis_t)X);
+ setAxisPosition_mm((Y_MIN_BED + lfrb[1]), (axis_t)Y);
+ waitway = 6;
+ break;
+ }
+ case 8: { // Assitant Level , Front Right 3
+ setAxisPosition_mm(BED_TRAMMING_Z_HOP, (axis_t)Z);
+ setAxisPosition_mm((X_MAX_BED - lfrb[2]), (axis_t)X);
+ setAxisPosition_mm((Y_MIN_BED + lfrb[1]), (axis_t)Y);
+ waitway = 6;
+ break;
+ }
+ case 9: { // Assitant Level , Back Right 4
+ setAxisPosition_mm(BED_TRAMMING_Z_HOP, (axis_t)Z);
+ setAxisPosition_mm((X_MAX_BED - lfrb[2]), (axis_t)X);
+ setAxisPosition_mm((Y_MAX_BED - lfrb[3]), (axis_t)Y);
+ waitway = 6;
+ break;
+ }
+ case 10: { // Assitant Level , Back Left 5
+ setAxisPosition_mm(BED_TRAMMING_Z_HOP, (axis_t)Z);
+ setAxisPosition_mm((X_MIN_BED + lfrb[0]), (axis_t)X);
+ setAxisPosition_mm((Y_MAX_BED - lfrb[3]), (axis_t)Y);
+ waitway = 6;
+ break;
+ }
+ case 11: { // Autolevel switch
+ #if HAS_MESH
+ if (!getLevelingActive()) { // turn on the Autolevel
+ RTS_SndData(3, AutoLevelIcon);
+ setLevelingActive(true);
+ }
+ else { // turn off the Autolevel
+ RTS_SndData(2, AutoLevelIcon);
+ setLevelingActive(false);
+ }
+ #endif
+ RTS_SndData(getZOffset_mm() * 100, ProbeOffset_Z);
+ break;
+ }
+ case 12: {
+ injectCommands(F("G26R255"));
+ onStatusChanged(F("Beginning G26.. Heating"));
+ break;
+ }
+ case 13: {
+ injectCommands(F("G29S1"));
+ onStatusChanged(F("Begin Manual Mesh"));
+ break;
+ }
+ case 14: {
+ injectCommands(F("G29S2"));
+ onStatusChanged(F("Moving to Next Mesh Point"));
+ break;
+ }
+ case 15: {
+ injectCommands(F("M211S0\nG91\nG1Z-0.025\nG90\nM211S1"));
+ onStatusChanged(F("Moved down 0.025"));
+ break;
+ }
+ case 16: {
+ injectCommands(F("M211S0\nG91\nG1Z0.025\nG90\nM211S1"));
+ onStatusChanged(F("Moved up 0.025"));
+ break;
+ }
+ case 17: {
+ Settings.display_volume = 0;
+ Settings.display_sound = false;
+ SetTouchScreenConfiguration();
+ break;
+ }
+ case 18: {
+ Settings.display_volume = 255;
+ Settings.display_sound = true;
+ SetTouchScreenConfiguration();
+ break;
+ }
+ case 19: {
+ Settings.screen_brightness = 10;
+ SetTouchScreenConfiguration();
+ break;
+ }
+ case 20: {
+ Settings.screen_brightness = 100;
+ SetTouchScreenConfiguration();
+ break;
+ }
+ case 21: {
+ Settings.display_standby ^= true;
+ SetTouchScreenConfiguration();
+ break;
+ }
+ case 22: {
+ Settings.screen_rotation = Settings.screen_rotation == 10 ? 0 : 10;
+ SetTouchScreenConfiguration();
+ break;
+ }
+ case 23: { // Set IDEX Autopark
+ injectCommands(F("M605S1\nG28X\nG1X0"));
+ break;
+ }
+ case 24: { // Set IDEX Duplication
+ injectCommands(F("M605S1\nT0\nG28\nM605S2\nG28X\nG1X0"));
+ break;
+ }
+ case 25: { // Set IDEX Mirrored Duplication
+ injectCommands(F("M605S1\nT0\nG28\nM605S2\nG28X\nG1X0\nM605S3"));
+ break;
+ }
+ case 26: { // Set IDEX Full Control
+ injectCommands(F("M605S0\nG28X"));
+ break;
+ }
+ case 27: { // Change Tool
+ setActiveTool(getActiveTool() == E0 ? E1 : E0, !isAxisPositionKnown(X));
+ break;
+ }
+ default: {
+ DEBUG_ECHOLNPGM("Unsupported Option Selected", recdat.data[0]);
+ break;
+ }
+ }
+
+ RTS_SndData(10, FilenameIcon);
+ break;
+
+ case XYZEaxis: {
+ axis_t axis = X;
+ float min = 0.0f, max = 0.0f;
+ waitway = 4;
+ if (recdat.addr == DisplayXaxis) {
+ axis = X;
+ min = X_MIN_POS;
+ max = X_MAX_POS;
+ }
+ else if (recdat.addr == DisplayYaxis) {
+ axis = Y;
+ min = Y_MIN_POS;
+ max = Y_MAX_POS;
+ }
+ else if (recdat.addr == DisplayZaxis) {
+ axis = Z;
+ min = Z_MIN_POS;
+ max = Z_MAX_POS;
+ }
+ else if (recdat.addr == AutoZero) {
+ if (recdat.data[0] == 3) { // autohome
+ waitway = 4;
+ injectCommands(F("G28\nG1 F1000 Z10"));
+ InforShowStatus = AutohomeKey = true;
+ AutoHomeIconNum = 0;
+ RTS_SndData(10, FilenameIcon);
+ }
+ else {
+ AxisPagenum = recdat.data[0];
+ waitway = 0;
+ }
+ break;
+ }
+
+ float targetPos = ((float)recdat.data[0]) / 10;
+ LIMIT(targetPos, min, max);
+ setAxisPosition_mm(targetPos, axis);
+ waitway = 0;
+ RTS_SndData(10, FilenameIcon);
+ break;
+ }
+
+ case Filament:
+
+ unsigned int IconTemp;
+ if (recdat.addr == Exchfilement) {
+ if (getActualTemp_celsius(getActiveTool()) < EXTRUDE_MINTEMP && recdat.data[0] < 5) {
+ RTS_SndData((int)EXTRUDE_MINTEMP, 0x1020);
+ delay_ms(5);
+ RTS_SndData(ExchangePageBase + 66, ExchangepageAddr);
+ break;
+ }
+
+ switch (recdat.data[0]) {
+ case 1: { // Unload filement1
+ setAxisPosition_mm((getAxisPosition_mm(E0) - ChangeMaterialbuf[0]), E0);
+ break;
+ }
+ case 2: { // Load filement1
+ setAxisPosition_mm((getAxisPosition_mm(E0) + ChangeMaterialbuf[0]), E0);
+ break;
+ }
+ case 3: { // Unload filement2
+ setAxisPosition_mm((getAxisPosition_mm(E1) - ChangeMaterialbuf[1]), E1);
+ break;
+ }
+ case 4: { // Load filement2
+ setAxisPosition_mm((getAxisPosition_mm(E1) + ChangeMaterialbuf[1]), E1);
+ break;
+ }
+ case 5: { // sure to heat
+ NozzleTempStatus[0] = 1;
+ // InforShowoStatus = true;
+
+ setTargetTemp_celsius((PREHEAT_1_TEMP_HOTEND + 10), getActiveTool());
+ IconTemp = getActualTemp_celsius(getActiveTool()) * 100 / getTargetTemp_celsius(getActiveTool());
+ if (IconTemp >= 100)
+ IconTemp = 100;
+ RTS_SndData(IconTemp, HeatPercentIcon);
+
+ RTS_SndData(getActualTemp_celsius(H0), NozzleTemp);
+ RTS_SndData(getTargetTemp_celsius(H0), NozzlePreheat);
+ #if HAS_MULTI_HOTEND
+ rtscheck.RTS_SndData(getActualTemp_celsius(H1), e2Temp);
+ rtscheck.RTS_SndData(getTargetTemp_celsius(H1), e2Preheat);
+ #else
+ rtscheck.RTS_SndData(0, e2Temp);
+ rtscheck.RTS_SndData(0, e2Preheat);
+ #endif
+ delay_ms(5);
+ RTS_SndData(ExchangePageBase + 68, ExchangepageAddr);
+ break;
+ }
+ case 6: { // cancel to heat
+ RTS_SndData(ExchangePageBase + 65, ExchangepageAddr);
+ break;
+ }
+ case 0xF1: { // Sure to cancel heating
+ // InforShowoStatus = true;
+ NozzleTempStatus[0] = 0;
+ delay_ms(1);
+ RTS_SndData(ExchangePageBase + 65, ExchangepageAddr);
+ break;
+ }
+ case 0xF0: // not to cancel heating
+ break;
+ }
+ RTS_SndData(10 * ChangeMaterialbuf[0], FilamentUnit1); // It's ChangeMaterialbuf for show,instead of current_position[E_AXIS] in them.
+ RTS_SndData(10 * ChangeMaterialbuf[1], FilamentUnit2);
+ }
+ else if (recdat.addr == FilamentUnit1) {
+ ChangeMaterialbuf[0] = ((float)recdat.data[0]) / 10;
+ }
+ else if (recdat.addr == FilamentUnit2) {
+ ChangeMaterialbuf[1] = ((float)recdat.data[0]) / 10;
+ }
+ break;
+
+ case LanguageChoice:
+
+ DEBUG_ECHOLNPGM("\n ***recdat.data[0] =", recdat.data[0]);
+ /*if(recdat.data[0]==1) {
+ settings.save();
+ }
+ else {
+ injectCommands(F("M300"));
+ }*/
+ // may at some point use language change screens to save eeprom explicitly
+ DEBUG_ECHOLNPGM("InLangChoice");
+ switch (recdat.data[0]) {
+ case 0: {
+ DEBUG_ECHOLNPGM("Store Settings");
+ injectCommands(F("M500"));
+ break;
+ }
+ case 1: {
+ RTS_SndData(ExchangePageBase + 94, ExchangepageAddr);
+ break;
+ }
+
+ #if ENABLED(PIDTEMP)
+ case 2: {
+ onStatusChanged(F("Hotend PID Started"));
+ startPIDTune(static_cast(pid_hotendAutoTemp), getActiveTool());
+ break;
+ }
+ #endif
+
+ case 3: {
+ DEBUG_ECHOLNPGM("Init EEPROM");
+ injectCommands(F("M502\nM500"));
+ break;
+ }
+ case 4: {
+ DEBUG_ECHOLNPGM("BLTouch Reset");
+ injectCommands(F("M999\nM280P0S160"));
+ break;
+ }
+
+ case 5: {
+ #if ENABLED(PIDTEMPBED)
+ onStatusChanged(F("Bed PID Started"));
+ startBedPIDTune(static_cast(pid_bedAutoTemp));
+ #else
+ DEBUG_ECHOLNPGM("Bed PID Disabled");
+ #endif
+ break;
+ }
+ case 6: {
+ DEBUG_ECHOLNPGM("Store Settings");
+ injectCommands(F("M500"));
+ break;
+ }
+ default: {
+ DEBUG_ECHOLNPGM("Invalid Option");
+ break;
+ }
+ }
+ break;
+ case No_Filament:
+ DEBUG_ECHOLNPGM("\n No Filament");
+
+ if (recdat.data[0] == 1) { // Filament is out, resume / resume selected on screen
+ DEBUG_ECHOLNPGM("Resume Yes during print");
+ if (ExtUI::pauseModeStatus != PAUSE_MESSAGE_PURGE && ExtUI::pauseModeStatus != PAUSE_MESSAGE_OPTION) {
+ // setPauseMenuResponse(PAUSE_RESPONSE_RESUME_PRINT);
+ setUserConfirmed();
+ // PrinterStatusKey[1] = 3;
+ // pause_resume_selected = true;
+ }
+ else if (ExtUI::pauseModeStatus == PAUSE_MESSAGE_PURGE || ExtUI::pauseModeStatus == PAUSE_MESSAGE_OPTION) {
+ #if ENABLED(FILAMENT_RUNOUT_SENSOR)
+ if (getFilamentRunoutState() && getFilamentRunoutEnabled(getActiveTool())) {
+ ExtUI::setFilamentRunoutEnabled(false, getActiveTool());
+ }
+ else {
+ setPauseMenuResponse(PAUSE_RESPONSE_RESUME_PRINT);
+ setUserConfirmed();
+ PrinterStatusKey[1] = 3;
+ pause_resume_selected = true;
+ }
+
+ #else
+ setPauseMenuResponse(PAUSE_RESPONSE_RESUME_PRINT);
+ setUserConfirmed();
+ PrinterStatusKey[1] = 3;
+ pause_resume_selected = true;
+ #endif
+ }
+ }
+ else if (recdat.data[0] == 0) { // Filamet is out, Cancel Selected
+ DEBUG_ECHOLNPGM(" Filament Response No");
+ if (ExtUI::pauseModeStatus == PAUSE_MESSAGE_PURGE || ExtUI::pauseModeStatus == PAUSE_MESSAGE_OPTION) {
+ setPauseMenuResponse(PAUSE_RESPONSE_EXTRUDE_MORE);
+ setUserConfirmed();
+ }
+ }
+ break;
+
+ #if ENABLED(POWER_LOSS_RECOVERY)
+ case PwrOffNoF:
+ if (recdat.data[0] == 1) // Yes:continue to print the 3Dmode during power-off.
+ injectCommands(F("M1000"));
+ else if (recdat.data[0] == 2) // No
+ injectCommands(F("M1000C"));
+ break;
+ #endif
+ case Volume:
+ if (recdat.data[0] < 0)
+ Settings.display_volume = 0;
+ else if (recdat.data[0] > 255)
+ Settings.display_volume = 0xFF;
+ else
+ Settings.display_volume = recdat.data[0];
+
+ if (Settings.display_volume == 0) {
+ RTS_SndData(0, VolumeIcon);
+ RTS_SndData(9, SoundIcon);
+ }
+ else {
+ RTS_SndData((Settings.display_volume + 1) / 32 - 1, VolumeIcon);
+ RTS_SndData(8, SoundIcon);
+ }
+ RTS_SndData(Settings.display_volume << 8, SoundAddr + 1);
+ break;
+
+ case Filename:
+ DEBUG_ECHOLNPGM("Filename Selected");
+ if (isMediaInserted() && recdat.addr == FilenameChs) {
+ DEBUG_ECHOLNPGM("Has Media");
+
+ recordcount = recdat.data[0] - 1;
+ if (filenavigator.currentindex == 0 && filenavigator.folderdepth > 0 && (fileIndex + recordcount) == 0) {
+ filenavigator.upDIR();
+ DEBUG_ECHOLNPGM("GoUpDir");
+ filenavigator.getFiles(0);
+ fileIndex = 0;
+ return;
+ }
+
+ if (filenavigator.currentindex == 0 && filenavigator.folderdepth > 0)
+ recordcount = recordcount - 1; // account for return dir link in file index
+
+ for (int j = 1; j <= 4; j++) { // Clear filename BG Color and Frame
+ RTS_SndData((unsigned long)0xFFFF, FilenameNature + j * 16); // white
+ RTS_SndData(10, FilenameIcon1 + j); // clean
+ }
+ for (int j = 0; j < 10; j++) // clear current filename
+ RTS_SndData(0, Choosefilename + j);
+
+ if (filenavigator.getIndexisDir(fileIndex + recordcount)) {
+ DEBUG_ECHOLNPGM("Is Dir ", (fileIndex + recordcount));
+ filenavigator.changeDIR((char *)filenavigator.getIndexName(fileIndex + recordcount));
+ filenavigator.getFiles(0);
+ fileIndex = 0;
+ return;
+ }
+ else {
+ DEBUG_ECHOLNPGM("Is File ", (fileIndex + recordcount));
+ RTS_SndData(filenavigator.getIndexName(fileIndex + recordcount), Choosefilename);
+ RTS_SndData((unsigned long)0x87F0, FilenameNature + recdat.data[0] * 16); // Change BG of selected line to Light Green
+ RTS_SndData(6, FilenameIcon1 + recdat.data[0]); // show frame
+ }
+ }
+ else if (recdat.addr == FilenamePlay) {
+ if (recdat.data[0] == 1 && isMediaInserted()) { // for sure
+ printFile(filenavigator.getIndexName(fileIndex + recordcount));
+
+ for (int j = 0; j < 10; j++) // clean screen.
+ RTS_SndData(0, Printfilename + j);
+
+ RTS_SndData(filenavigator.getIndexName(fileIndex + recordcount), Printfilename);
+ delay_ms(2);
+
+ delay_ms(2);
+ RTS_SndData(ExchangePageBase + 53, ExchangepageAddr);
+
+ TPShowStatus = InforShowStatus = true;
+ PrinterStatusKey[0] = 1;
+ PrinterStatusKey[1] = 3;
+ fileIndex = 0;
+ recordcount = 0;
+ }
+ else if (recdat.data[0] == 2) { // Page Down
+ DEBUG_ECHOLNPGM("PgDown");
+ if ((fileIndex + DISPLAY_FILES) < (filenavigator.maxFiles() + (filenavigator.folderdepth != 0))) {
+ fileIndex = fileIndex + DISPLAY_FILES;
+ // if(filenavigator.folderdepth!=0 && fileIndex!=0) //Shift to acknowledge Return DIR button on first page
+ // filenavigator.getFiles(fileIndex-1);
+ // else
+ filenavigator.getFiles(fileIndex);
+ // filenavigator.getFiles(filenavigator.currentindex+1);
+ }
+ }
+ else if (recdat.data[0] == 3) { // Page Up
+ DEBUG_ECHOLNPGM("PgUp");
+ if (fileIndex >= DISPLAY_FILES) {
+ fileIndex = fileIndex - DISPLAY_FILES;
+ // if(filenavigator.folderdepth!=0 && fileIndex!=0) //Shift to acknowledge Return DIR button on first page
+ // filenavigator.getFiles(filenavigator.currentindex-DISPLAY_FILES);
+ // else
+ filenavigator.getFiles(fileIndex);
+ }
+ }
+ else if (recdat.data[0] == 4) { // Page Up
+ DEBUG_ECHOLNPGM("Refresh");
+ injectCommands(F("M22\nM21"));
+ }
+ else if (recdat.data[0] == 0) { // return to main page
+ InforShowStatus = true;
+ TPShowStatus = false;
+ }
+ }
+ break;
+
+ case VolumeDisplay: {
+ DEBUG_ECHOLNPGM("VolumeDisplay");
+ if (recdat.data[0] == 0) {
+ Settings.display_volume = 0;
+ Settings.display_sound = false;
+ }
+ else if (recdat.data[0] > 100) {
+ Settings.display_volume = 255;
+ Settings.display_sound = true;
+ }
+ else {
+ Settings.display_volume = (uint8_t)map(constrain(recdat.data[0], 0, 100), 0, 100, 0, 255);
+ Settings.display_sound = true;
+ }
+ SetTouchScreenConfiguration();
+ break;
+ }
+
+ case DisplayBrightness: {
+ DEBUG_ECHOLNPGM("DisplayBrightness LCD: ", recdat.data[0]);
+ if (recdat.data[0] < 10)
+ Settings.screen_brightness = 10;
+ else if (recdat.data[0] > 100)
+ Settings.screen_brightness = 100;
+ else
+ Settings.screen_brightness = (uint8_t)recdat.data[0];
+ DEBUG_ECHOLNPGM("DisplayBrightness Set: ", Settings.screen_brightness);
+ SetTouchScreenConfiguration();
+ break;
+ }
+
+ case DisplayStandbyBrightness: {
+ DEBUG_ECHOLNPGM("DisplayStandbyBrightness");
+ if (recdat.data[0] < 10)
+ Settings.standby_screen_brightness = 10;
+ else if (recdat.data[0] > 100)
+ Settings.standby_screen_brightness = 100;
+ else
+ Settings.standby_screen_brightness = (uint8_t)recdat.data[0];
+ SetTouchScreenConfiguration();
+ break;
+ }
+
+ case DisplayStandbySeconds: {
+ DEBUG_ECHOLNPGM("DisplayStandbySeconds");
+ if (recdat.data[0] < 5)
+ Settings.standby_time_seconds = 5;
+ else if (recdat.data[0] > 100)
+ Settings.standby_time_seconds = 100;
+ else
+ Settings.standby_time_seconds = (uint8_t)recdat.data[0];
+ SetTouchScreenConfiguration();
+ break;
+ }
+
+ case AutolevelVal:
+ {
+ uint8_t meshPoint = (recdat.addr - AutolevelVal) / 2;
+ uint8_t yPnt = floor(meshPoint / GRID_MAX_POINTS_X);
+ uint8_t xPnt;
+ if (yPnt % 2 == 0)
+ xPnt = meshPoint - (yPnt * GRID_MAX_POINTS_X);
+ else
+ xPnt = (GRID_MAX_POINTS_X - 1) - (meshPoint - (yPnt * GRID_MAX_POINTS_X)); // zag row
+ float meshVal;
+ DEBUG_ECHOLNPGM("meshPoint ", meshPoint);
+ DEBUG_ECHOLNPGM("xPnt ", xPnt);
+ DEBUG_ECHOLNPGM("yPnt ", yPnt);
+
+ if (recdat.data[0] >= 32768)
+ meshVal = ((float)recdat.data[0] - 65536) / 1000;
+ else
+ meshVal = ((float)recdat.data[0]) / 1000;
+
+ DEBUG_ECHOLNPGM("meshVal ", meshVal);
+ meshVal = constrain(meshVal, Z_PROBE_LOW_POINT, Z_CLEARANCE_BETWEEN_PROBES);
+ DEBUG_ECHOLNPGM("Constrain meshVal ", meshVal);
+ xy_uint8_t point = {xPnt, yPnt};
+ setMeshPoint(point, meshVal);
+ rtscheck.RTS_SndData((meshVal * 1000), recdat.addr);
+ break;
+ }
+
+ default:
+ DEBUG_ECHOLNPGM("No Match :");
+ break;
+ }
+
+ memset(&recdat, 0, sizeof(recdat));
+ recdat.head[0] = FHONE;
+ recdat.head[1] = FHTWO;
+ }
+
+ void WriteVariable(uint16_t adr, const void* values, uint8_t valueslen, bool isstr=false, char fillChar=' ') {
+ const char* myvalues = static_cast(values);
+ bool strend = !myvalues;
+ DWIN_SERIAL.write(FHONE);
+ DWIN_SERIAL.write(FHTWO);
+ DWIN_SERIAL.write(valueslen + 3);
+ DWIN_SERIAL.write(0x82);
+ DWIN_SERIAL.write(adr >> 8);
+ DWIN_SERIAL.write(adr & 0xFF);
+ while (valueslen--) {
+ char x;
+ if (!strend) x = *myvalues++;
+ if ((isstr && !x) || strend) {
+ strend = true;
+ x = fillChar;
+ }
+ DWIN_SERIAL.write(x);
+ }
+ }
+
+ void SetTouchScreenConfiguration() {
+ // Main configuration (System_Config)
+ LIMIT(Settings.screen_brightness, 10, 100); // Prevent a possible all-dark screen
+ LIMIT(Settings.standby_time_seconds, 10, 655); // Prevent a possible all-dark screen for standby, yet also don't go higher than the DWIN limitation
+
+
+ unsigned char cfg_bits = 0x0;
+ // #if ENABLED(DWINOS_4)
+ cfg_bits |= 1UL << 7; // 7: Enable Control
+ // #endif
+ cfg_bits |= 1UL << 5; // 5: load 22 touch file
+ cfg_bits |= 1UL << 4; // 4: auto-upload should always be enabled
+ if (Settings.display_sound) cfg_bits |= 1UL << 3; // 3: audio
+ if (Settings.display_standby) cfg_bits |= 1UL << 2; // 2: backlight on standby
+ if (Settings.screen_rotation == 10) cfg_bits |= 1UL << 1; // 1 & 0: Inversion
+ #if ANY(MachineCR10Smart, MachineCR10SmartPro )
+ cfg_bits |= 1UL << 0; // Portrait Mode or 800x480 display has 0 point rotated 90deg from 480x272 display
+ #endif
+
+
+
+ const unsigned char config_set[] = {
+ #if ENABLED(DWINOS_4)
+ 0x5A, 0x00, (unsigned char)(cfg_bits >> 8U), (unsigned char)(cfg_bits & 0xFFU)
+ #else
+ 0x5A, 0x00, 0xFF, cfg_bits
+ #endif
+ };
+ WriteVariable(0x80 /*System_Config*/, config_set, sizeof(config_set));
+
+ // Standby brightness (LED_Config)
+ uint16_t dwinStandbyTimeSeconds = 100 * Settings.standby_time_seconds; /* milliseconds, but divided by 10 (not 5 like the docs say) */
+ const unsigned char brightness_set[] = {
+ Settings.screen_brightness /*% active*/,
+ Settings.standby_screen_brightness /*% standby*/,
+ static_cast(dwinStandbyTimeSeconds >> 8),
+ static_cast(dwinStandbyTimeSeconds)
+ };
+ WriteVariable(0x82 /*LED_Config*/, brightness_set, sizeof(brightness_set));
+
+ if (!Settings.display_sound) {
+ rtscheck.RTS_SndData(0, VolumeIcon);
+ rtscheck.RTS_SndData(9, SoundIcon);
+ }
+ else {
+ rtscheck.RTS_SndData((Settings.display_volume + 1) / 32 - 1, VolumeIcon);
+ rtscheck.RTS_SndData(8, SoundIcon);
+ }
+ rtscheck.RTS_SndData(Settings.display_volume, VolumeIcon - 2);
+ rtscheck.RTS_SndData(Settings.display_volume << 8, SoundAddr + 1);
+ rtscheck.RTS_SndData(map(constrain(Settings.display_volume, 0, 255), 0, 255, 0, 100), VolumeDisplay);
+ rtscheck.RTS_SndData(Settings.screen_brightness, DisplayBrightness);
+ rtscheck.RTS_SndData(Settings.standby_screen_brightness, DisplayStandbyBrightness);
+ rtscheck.RTS_SndData(Settings.standby_time_seconds, DisplayStandbySeconds);
+ if (Settings.display_standby)
+ rtscheck.RTS_SndData(3, DisplayStandbyEnableIndicator);
+ else
+ rtscheck.RTS_SndData(2, DisplayStandbyEnableIndicator);
+ }
+
+ void onPrinterKilled(FSTR_P const error, FSTR_P const component) {
+ DEBUG_ECHOLNPGM("***kill***");
+ rtscheck.RTS_SndData(ExchangePageBase + 88, ExchangepageAddr);
+ int j = 0;
+ char outmsg[40];
+ char killMsg[strlen_P(FTOP(error)) + strlen_P(FTOP(component)) + 3];
+ sprintf_P(killMsg, PSTR(S_FMT ": " S_FMT), FTOP(error), FTOP(component));
+ while (j < 4) {
+ outmsg[j] = '*';
+ j++;
+ }
+ while (const char c = killMsg[j - 4]) {
+ outmsg[j] = c;
+ j++;
+ }
+ while (j < 40) {
+ outmsg[j] = '*';
+ j++;
+ }
+ rtscheck.RTS_SndData(outmsg, MacVersion);
+ delay_ms(10);
+ }
+
+ void onMediaInserted() {
+ DEBUG_ECHOLNPGM("***Initing card is OK***");
+ filenavigator.reset();
+ filenavigator.getFiles(0);
+ fileIndex = 0;
+ recordcount = 0;
+ }
+
+ void onMediaError() {
+ filenavigator.reset();
+ for (int i = 0; i < MaxFileNumber; i++)
+ for (int j = 0; j < 10; j++) rtscheck.RTS_SndData(0, SDFILE_ADDR + i * 10 + j);
+
+ for (int j = 0; j < 10; j++) {
+ rtscheck.RTS_SndData(0, Printfilename + j); // clean screen.
+ rtscheck.RTS_SndData(0, Choosefilename + j); // clean filename
+ }
+ for (int j = 0; j < 8; j++) rtscheck.RTS_SndData(0, FilenameCount + j);
+ for (int j = 1; j <= MaxFileNumber; j++) {
+ rtscheck.RTS_SndData(10, FilenameIcon + j);
+ rtscheck.RTS_SndData(10, FilenameIcon1 + j);
+ }
+ return;
+ DEBUG_ECHOLNPGM("***Initing card fails***");
+ }
+
+ void onMediaRemoved() {
+ filenavigator.reset();
+ for (int i = 0; i < MaxFileNumber; i++)
+ for (int j = 0; j < 10; j++) rtscheck.RTS_SndData(0, SDFILE_ADDR + i * 10 + j);
+
+ for (int j = 0; j < 10; j++) {
+ rtscheck.RTS_SndData(0, Printfilename + j); // clean screen.
+ rtscheck.RTS_SndData(0, Choosefilename + j); // clean filename
+ }
+ for (int j = 0; j < 8; j++) rtscheck.RTS_SndData(0, FilenameCount + j);
+ for (int j = 1; j <= MaxFileNumber; j++) {
+ rtscheck.RTS_SndData(10, FilenameIcon + j);
+ rtscheck.RTS_SndData(10, FilenameIcon1 + j);
+ }
+ return;
+ DEBUG_ECHOLNPGM("***Card Removed***");
+ }
+
+ void onPlayTone(const uint16_t frequency, const uint16_t duration) {
+ DEBUG_ECHOLNPGM("***CPlay Tone***");
+ rtscheck.RTS_SndData(StartSoundSet, SoundAddr);
+ }
+
+ void onPrintTimerStarted() {
+ DEBUG_ECHOLNPGM("==onPrintTimerStarted==");
+ if (waitway == 7)
+ return;
+ PrinterStatusKey[1] = 3;
+ InforShowStatus = true;
+ delay_ms(1);
+ rtscheck.RTS_SndData(ExchangePageBase + 53, ExchangepageAddr);
+ }
+
+ void onPrintTimerPaused() {
+ DEBUG_ECHOLNPGM("==onPrintTimerPaused==");
+ rtscheck.RTS_SndData(ExchangePageBase + 78, ExchangepageAddr); // Display Pause Screen
+ onStatusChanged(F("Pausing..."));
+ }
+ void onPrintTimerStopped() {
+ DEBUG_ECHOLNPGM("==onPrintTimerStopped==");
+ if (waitway == 3) return;
+
+ #if FAN_COUNT > 0
+ for (uint8_t i = 0; i < FAN_COUNT; i++) setTargetFan_percent(FanOff, (fan_t)i);
+ #endif
+
+ PrinterStatusKey[0] = 0;
+ InforShowStatus = true;
+ TPShowStatus = false;
+ rtscheck.RTS_SndData(ExchangePageBase + 51, ExchangepageAddr);
+ }
+
+ void onFilamentRunout() {
+ DEBUG_ECHOLNPGM("==onFilamentRunout==");
+ PrinterStatusKey[1] = 4;
+ TPShowStatus = false;
+ rtscheck.RTS_SndData(ExchangePageBase + 78, ExchangepageAddr);
+ }
+ void onFilamentRunout(extruder_t extruder) {
+ DEBUG_ECHOLNPGM("==onFilamentRunout==");
+ PrinterStatusKey[1] = 4;
+ TPShowStatus = false;
+ rtscheck.RTS_SndData(ExchangePageBase + 78, ExchangepageAddr);
+ }
+ void onUserConfirmRequired(const char *const msg) {
+ PrinterStatusKey[1] = 4;
+ TPShowStatus = false;
+ if (lastPauseMsgState == ExtUI::pauseModeStatus && msg == (const char*)GET_TEXT_F(MSG_FILAMENT_CHANGE_LOAD))
+ return;
+
+ switch (ExtUI::pauseModeStatus) {
+ case PAUSE_MESSAGE_WAITING: {
+ rtscheck.RTS_SndData(ExchangePageBase + 78, ExchangepageAddr);
+ onStatusChanged(F("Press Yes to Continue"));
+ break;
+ }
+ case PAUSE_MESSAGE_INSERT: {
+ rtscheck.RTS_SndData(ExchangePageBase + 78, ExchangepageAddr);
+ onStatusChanged(F("Load Filament to Continue"));
+ break;
+ }
+ case PAUSE_MESSAGE_HEAT: {
+ rtscheck.RTS_SndData(ExchangePageBase + 78, ExchangepageAddr);
+ onStatusChanged(F("Add Filament and Press Yes to Reheat"));
+ break;
+ }
+
+ #if DISABLED(ADVANCED_PAUSE_CONTINUOUS_PURGE)
+ case PAUSE_MESSAGE_PURGE: {
+ rtscheck.RTS_SndData(ExchangePageBase + 78, ExchangepageAddr);
+ char newMsg[40] = "Yes to ";
+ if (TERN1(FILAMENT_RUNOUT_SENSOR, (!ExtUI::getFilamentRunoutState() && getFilamentRunoutEnabled())))
+ strcat_P(newMsg, PSTR("Continue"));
+ else
+ strcat_P(newMsg, PSTR("Disable "));
+
+ strcat_P(newMsg, PSTR(" No to Purge"));
+ onStatusChanged(newMsg);
+ break;
+ }
+ #endif
+
+ case PAUSE_MESSAGE_OPTION: {
+ rtscheck.RTS_SndData(ExchangePageBase + 78, ExchangepageAddr);
+ char newMsg[40] = "Yes to ";
+ if (TERN1(FILAMENT_RUNOUT_SENSOR, (!ExtUI::getFilamentRunoutState() && getFilamentRunoutEnabled())))
+ strcat_P(newMsg, PSTR("Continue"));
+ else
+ strcat_P(newMsg, PSTR("Disable "));
+
+ strcat_P(newMsg, PSTR(" No to Purge"));
+ onStatusChanged(newMsg);
+ break;
+ }
+
+ case PAUSE_MESSAGE_PARKING: {
+ rtscheck.RTS_SndData(ExchangePageBase + 78, ExchangepageAddr);
+ onStatusChanged(F("Parking..."));
+ break;
+ }
+ case PAUSE_MESSAGE_CHANGING: {
+ rtscheck.RTS_SndData(ExchangePageBase + 78, ExchangepageAddr);
+ onStatusChanged(F("Beginning Filament Change"));
+ break;
+ }
+ case PAUSE_MESSAGE_UNLOAD: {
+ rtscheck.RTS_SndData(ExchangePageBase + 78, ExchangepageAddr);
+ onStatusChanged(F("Unloading..."));
+ break;
+ }
+ case PAUSE_MESSAGE_LOAD: {
+ rtscheck.RTS_SndData(ExchangePageBase + 78, ExchangepageAddr);
+ onStatusChanged(F("Reloading..."));
+ break;
+ }
+ case PAUSE_MESSAGE_RESUME:
+ #if ENABLED(ADVANCED_PAUSE_CONTINUOUS_PURGE)
+ case PAUSE_MESSAGE_PURGE: {
+ rtscheck.RTS_SndData(ExchangePageBase + 78, ExchangepageAddr);
+ onStatusChanged(F("Press Yes to Stop Purge"));
+ break;
+ }
+ #endif
+
+ case PAUSE_MESSAGE_HEATING: {
+ rtscheck.RTS_SndData(ExchangePageBase + 68, ExchangepageAddr);
+ onStatusChanged(F("Reheating"));
+ break;
+ }
+
+ case PAUSE_MESSAGE_STATUS:
+ default: {
+ setPauseMenuResponse(PAUSE_RESPONSE_RESUME_PRINT);
+ setUserConfirmed();
+ DEBUG_ECHOLNPGM("Pause Mode Status");
+ break;
+ }
+ }
+ lastPauseMsgState = ExtUI::pauseModeStatus;
+ DEBUG_ECHOLNPGM("==onUserConfirmRequired==", pauseModeStatus);
+ }
+
+ void onStatusChanged(const char *const statMsg) {
+ for (int j = 0; j < 20; j++) // Clear old message
+ rtscheck.RTS_SndData(' ', StatusMessageString + j);
+ rtscheck.RTS_SndData(statMsg, StatusMessageString);
+ }
+ void onFactoryReset() {
+ Settings.settings_size = sizeof(creality_dwin_settings_t);
+ Settings.settings_version = dwin_settings_version;
+ Settings.display_standby = true;
+ Settings.display_sound = true;
+ Settings.display_volume = 32;
+ Settings.standby_screen_brightness = 15;
+ Settings.screen_brightness = 100;
+ Settings.standby_time_seconds = 60;
+ Settings.screen_rotation = 0;
+ onStartup();
+ startprogress = 0;
+ InforShowStatus = true;
+ DEBUG_ECHOLNPGM("==onFactoryReset==");
+ }
+
+ void onMeshUpdate(const int8_t xpos, const int8_t ypos, probe_state_t state) {}
+
+ void onMeshUpdate(const int8_t xpos, const int8_t ypos, const_float_t zval) {
+ if (waitway == 3)
+ if (isPositionKnown() && (getActualTemp_celsius(BED) >= (getTargetTemp_celsius(BED) - 1)))
+ rtscheck.RTS_SndData(ExchangePageBase + 64, ExchangepageAddr);
+ #if HAS_MESH
+ uint8_t abl_probe_index = 0;
+ for (uint8_t outer = 0; outer < GRID_MAX_POINTS_Y; outer++)
+ for (uint8_t inner = 0; inner < GRID_MAX_POINTS_X; inner++) {
+ const bool zig = (outer & 1); // != ((PR_OUTER_END) & 1);
+ const xy_uint8_t point = { zig ? (GRID_MAX_POINTS_X - 1) - inner : inner, outer };
+ if (x_Point == xpos && outer == ypos)
+ rtscheck.RTS_SndData(ExtUI::getMeshPoint(point) * 1000, AutolevelVal + (abl_probe_index * 2));
+ ++abl_probe_index;
+ }
+ #endif
+ }
+
+ void onStoreSettings(char *buff) {
+ static_assert(
+ ExtUI::eeprom_data_size >= sizeof(creality_dwin_settings_t),
+ "Insufficient space in EEPROM for UI parameters"
+ );
+
+ // Write to buffer
+ DEBUG_ECHOLNPGM("Saving DWIN LCD setting from EEPROM");
+ memcpy(buff, &Settings, sizeof(creality_dwin_settings_t));
+ }
+
+ void onLoadSettings(const char *buff) {
+ static_assert(
+ ExtUI::eeprom_data_size >= sizeof(creality_dwin_settings_t),
+ "Insufficient space in EEPROM for UI parameters"
+ );
+
+ creality_dwin_settings_t eepromSettings;
+ memcpy(&eepromSettings, buff, sizeof(creality_dwin_settings_t));
+
+ // If size is not the same, discard settings
+ if (eepromSettings.settings_size != sizeof(creality_dwin_settings_t)) {
+ DEBUG_ECHOLNPGM("Discarding DWIN LCD setting from EEPROM - size incorrect");
+ onFactoryReset();
+ return;
+ }
+
+ if (eepromSettings.settings_version != dwin_settings_version) {
+ DEBUG_ECHOLNPGM("Discarding DWIN LCD setting from EEPROM - settings version incorrect");
+ onFactoryReset();
+ return;
+ }
+
+ // Copy into final location
+ DEBUG_ECHOLNPGM("Loading DWIN LCD setting from EEPROM");
+ memcpy(&Settings, &eepromSettings, sizeof(creality_dwin_settings_t));
+
+ DEBUG_ECHOLNPGM("Setting Brightness : ", Settings.screen_brightness);
+ DEBUG_ECHOLNPGM("Setting Standby : ", Settings.standby_screen_brightness);
+ DEBUG_ECHOLNPGM("Setting Standby Time : ", Settings.standby_time_seconds);
+ DEBUG_ECHOLNPGM("Setting Rotation : ", Settings.screen_rotation);
+ DEBUG_ECHOLNPGM("Setting Volume : ", Settings.display_volume);
+
+ DEBUG_ECHOLNPGM("Setting Standby On : ", Settings.display_standby);
+ DEBUG_ECHOLNPGM("Setting Volume On : ", Settings.display_sound);
+
+ SetTouchScreenConfiguration();
+ }
+
+ void onSettingsStored(bool success) {
+ DEBUG_ECHOLNPGM("==onSettingsStored==");
+ // This is called after the entire EEPROM has been written,
+ // whether successful or not.
+ }
+
+ void onSettingsLoaded(bool success) {
+ DEBUG_ECHOLNPGM("==onConfigurationStoreRead==");
+ #if HAS_MESH
+ if (ExtUI::getMeshValid()) {
+ uint8_t abl_probe_index = 0;
+ for (uint8_t outer = 0; outer < GRID_MAX_POINTS_Y; outer++)
+ for (uint8_t inner = 0; inner < GRID_MAX_POINTS_X; inner++) {
+ const bool zig = (outer & 1);
+ const xy_uint8_t point = { zig ? (GRID_MAX_POINTS_X - 1) - inner : inner, outer };
+ rtscheck.RTS_SndData(ExtUI::getMeshPoint(point) * 1000, AutolevelVal + (abl_probe_index * 2));
+ ++abl_probe_index;
+ }
+
+ rtscheck.RTS_SndData(3, AutoLevelIcon); // 2=On, 3=Off
+ setLevelingActive(true);
+ }
+ else {
+ rtscheck.RTS_SndData(2, AutoLevelIcon); /*Off*/
+ setLevelingActive(false);
+ }
+ #endif
+
+ DEBUG_ECHOLNPGM("\n init zprobe_zoffset = ", getZOffset_mm());
+ rtscheck.RTS_SndData(getZOffset_mm() * 100, ProbeOffset_Z);
+ SetTouchScreenConfiguration();
+ }
+
+ #if ENABLED(POWER_LOSS_RECOVERY)
+ void onPowerLossResume() {
+ DEBUG_ECHOLNPGM("==OnPowerLossResume==");
+ startprogress = 254;
+ InforShowStatus = true;
+ TPShowStatus = false;
+ reEntryPrevent = false;
+ rtscheck.RTS_SndData(ExchangePageBase + 76, ExchangepageAddr);
+ }
+ #endif
+
+ #if HAS_PID_HEATING
+ void onPidTuning(const result_t rst) {
+ // Called for temperature PID tuning result
+ rtscheck.RTS_SndData(pid_hotendAutoTemp, HotendPID_AutoTmp);
+ rtscheck.RTS_SndData(pid_bedAutoTemp, BedPID_AutoTmp);
+ rtscheck.RTS_SndData((unsigned int)(getPID_Kp(E0) * 10), HotendPID_P);
+ rtscheck.RTS_SndData((unsigned int)(getPID_Ki(E0) * 10), HotendPID_I);
+ rtscheck.RTS_SndData((unsigned int)(getPID_Kd(E0) * 10), HotendPID_D);
+ #if ENABLED(PIDTEMPBED)
+ rtscheck.RTS_SndData((unsigned int)(getBedPID_Kp() * 10), BedPID_P);
+ rtscheck.RTS_SndData((unsigned int)(getBedPID_Ki() * 10), BedPID_I);
+ rtscheck.RTS_SndData((unsigned int)(getBedPID_Kd() * 10), BedPID_D);
+ #endif
+ onStatusChanged(F("PID Tune Finished"));
+ }
+ #endif
+
+ void onLevelingStart() {}
+
+ void onLevelingDone() {
+ #if HAS_MESH
+ if (ExtUI::getMeshValid()) {
+ uint8_t abl_probe_index = 0;
+ for (uint8_t outer = 0; outer < GRID_MAX_POINTS_Y; outer++)
+ for (uint8_t inner = 0; inner < GRID_MAX_POINTS_X; inner++) {
+ uint8_t x_Point = inner;
+ bool zig = (outer & 1);
+ if (zig) x_Point = (GRID_MAX_POINTS_X - 1) - inner;
+ xy_uint8_t point = {x_Point, outer};
+ rtscheck.RTS_SndData(ExtUI::getMeshPoint(point) * 1000, AutolevelVal + (abl_probe_index * 2));
+ ++abl_probe_index;
+ }
+
+ rtscheck.RTS_SndData(3, AutoLevelIcon); // 2=On, 3=Off
+ setLevelingActive(true);
+ }
+ else {
+ rtscheck.RTS_SndData(2, AutoLevelIcon); /*Off*/
+ setLevelingActive(false);
+ }
+ #endif
+ }
+
+ void onSteppersEnabled() {}
+ void onPrintDone() {}
+ void onHomingStart() {}
+ void onHomingDone() {}
+ void onSteppersDisabled() {}
+ void onPostprocessSettings() {}
+
+} // namespace ExtUI
+
+#endif // DGUS_LCD_UI_IA_CREALITY
diff --git a/Marlin/src/lcd/extui/ia_creality/creality_extui.h b/Marlin/src/lcd/extui/ia_creality/creality_extui.h
new file mode 100644
index 000000000000..a051ea92c2c8
--- /dev/null
+++ b/Marlin/src/lcd/extui/ia_creality/creality_extui.h
@@ -0,0 +1,303 @@
+/**
+ * Marlin 3D Printer Firmware
+ * Copyright (c) 2022 MarlinFirmware [https://github.com/MarlinFirmware/Marlin]
+ *
+ * Based on Sprinter and grbl.
+ * Copyright (c) 2011 Camiel Gubbels / Erik van der Zalm
+ *
+ * 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 3 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 .
+ *
+ */
+#pragma once
+
+/* ****************************************
+ * lcd/extui/ia_creality/creality_extui.h
+ * ****************************************
+ * Extensible_UI implementation for Creality DWIN
+ * 10SPro, Max, CRX, and others
+ * Based original Creality release, ported to ExtUI for Marlin 2.0
+ * Written by Insanity Automation, sponsored by Tiny Machines 3D
+ *
+ * ***************************************/
+
+#include "string.h"
+#include
+#include "../ui_api.h"
+
+/*********************************/
+#define FHONE (0x5A)
+#define FHTWO (0xA5)
+#define FHLENG (0x06)
+
+#define TEXTBYTELEN 18
+#define MaxFileNumber 20 // 16
+
+#define CEIconGrap 12
+#define FileNum MaxFileNumber
+#define FileNameLen TEXTBYTELEN
+
+#define SizeofDatabuf 46
+
+/*************Register and Variable addr*****************/
+#define RegAddr_W 0x80
+#define RegAddr_R 0x81
+#define VarAddr_W 0x82
+#define VarAddr_R 0x83
+#define ExchangePageBase (unsigned long)0x5A010000 // the first page ID. other page = first page ID + relevant num;
+#define StartSoundSet ((unsigned long)0x060480A0) // 06,start-music; 04, 4 musics; 80, the volume value; 04, return value about music number.
+//#define FONT_EEPROM 90
+//#define AutoLeve_EEPROM 100
+//#define FanOn 255
+#define FanOff 0
+
+/*variable addr*/
+#define ExchangepageAddr 0x0084
+#define SoundAddr 0x00A0
+#define StartIcon 0x1000
+#define FeedrateDisplay 0x1006 // Speed
+#define Stopprint 0x1008
+#define Pauseprint 0x100A
+#define Resumeprint 0x100C
+#define PrintscheduleIcon 0x100E
+#define Timehour 0x1010
+#define Timemin 0x1012
+#define IconPrintstatus 0x1014
+#define Percentage 0x1016
+#define FanKeyIcon 0x101E
+#define Flowrate 0x1300
+
+#define RunoutToggle 0x1018
+#define PowerLossToggle 0x101A
+#define LedToggle 0x101C
+
+#define StepMM_X 0x1242
+#define StepMM_Y 0x1246
+#define StepMM_Z 0x124A
+#define StepMM_E 0x124E
+
+#define ProbeOffset_X 0x1236
+#define ProbeOffset_Y 0x123A
+#define ProbeOffset_Z 0x1026
+
+#define T2Offset_X 0x1090
+#define T2Offset_Y 0x1092
+#define T2Offset_Z 0x1094
+#define T2StepMM_E 0x1096
+
+#define ActiveToolVP 0x1014
+
+#define HotendPID_AutoTmp 0x1252
+#define BedPID_AutoTmp 0x1254
+
+#define HotendPID_P 0x1256
+#define HotendPID_I 0x125A
+#define HotendPID_D 0x125E
+
+#define BedPID_P 0x1262
+#define BedPID_I 0x1266
+#define BedPID_D 0x126A
+
+#define Jerk_X 0x1270
+#define Jerk_Y 0x1272
+#define Jerk_Z 0x1274
+#define Jerk_E 0x1276
+
+#define Feed_X 0x1278
+#define Feed_Y 0x127A
+#define Feed_Z 0x127C
+#define Feed_E 0x127E
+
+#define Accel_X 0x1280
+#define Accel_Y 0x1282
+#define Accel_Z 0x1284
+#define Accel_E 0x1286
+
+#define HeatPercentIcon 0x1024
+
+#define NzBdSet 0x1032 // cooldown Hotend and Bed
+#define NozzlePreheat 0x1034 // setpoint
+#define NozzleTemp 0x1036 // Actual
+#define BedPreheat 0x103A // Setpoint
+#define Bedtemp 0x103C // Actual
+
+#define e2Temp 0x1050
+#define e2Preheat 0x104E
+
+#define AutoZeroIcon 0x1042
+#define AutoLevelIcon 0x1045
+#define AutoZero 0x1046
+#define DisplayXaxis 0x1048
+#define DisplayYaxis 0x104A
+#define DisplayZaxis 0x104C
+
+#define FilamentUnit1 0x1054
+#define Exchfilement 0x1056
+#define FilamentUnit2 0x1058
+
+#define MacVersion 0x1060
+#define SoftVersion 0x106A
+#define PrinterSize 0x1074
+#define CorpWebsite 0x107E
+#define VolumeIcon 0x108A
+#define SoundIcon 0x108C
+#define AutolevelIcon 0x108D
+#define ExchFlmntIcon 0x108E
+#define AutolevelVal 0x1100
+
+#define FilenameIcon 0x1200
+#define FilenameIcon1 0x1220
+#define Printfilename 0x2000
+#define FilesCurentPage 0x1310
+#define FilesMaxPage 0x1312
+#define SDFILE_ADDR 0x200A
+#define FilenamePlay 0x20D2
+#define FilenameChs 0x20D3
+#define Choosefilename 0x20D4
+#define FilenameCount 0x20DE
+#define FilenameNature 0x6003
+
+#define VolumeDisplay 0x1140
+#define DisplayBrightness 0x1142
+#define DisplayStandbyBrightness 0x1144
+#define DisplayStandbyEnableIndicator 0x1146
+#define DisplayStandbySeconds 0x1148
+
+#define StatusMessageString 0x2064
+
+#ifdef TARGET_STM32F4
+ #define DWIN_SERIAL Serial1
+#else
+ #define DWIN_SERIAL LCD_SERIAL
+#endif
+
+namespace ExtUI {
+
+ /************struct**************/
+
+ typedef enum : uint8_t {
+ DGUS_IDLE, //< waiting for DGUS_HEADER1.
+ DGUS_HEADER1_SEEN, //< DGUS_HEADER1 received
+ DGUS_HEADER2_SEEN, //< DGUS_HEADER2 received
+ DGUS_WAIT_TELEGRAM, //< LEN received, Waiting for to receive all bytes.
+ } rx_datagram_state_t;
+
+ typedef struct DataBuf {
+ unsigned char len;
+ unsigned char head[2];
+ unsigned char command;
+ unsigned long addr;
+ unsigned long bytelen;
+ unsigned short data[32];
+ unsigned char reserv[4];
+ } DB;
+
+ struct creality_dwin_settings_t {
+ size_t settings_size;
+ uint8_t settings_version;
+
+ bool display_standby;
+ bool display_sound;
+ int8_t screen_rotation;
+ int16_t display_volume;
+ uint8_t standby_screen_brightness;
+ uint8_t screen_brightness;
+ int16_t standby_time_seconds;
+ };
+
+ void SetTouchScreenConfiguration();
+
+ class RTSSHOW {
+ public:
+ RTSSHOW();
+ int RTS_RecData();
+ void RTS_SDCardInit(void);
+ void RTS_SDCardUpate(bool, bool);
+ int RTS_CheckFilament(int);
+ void RTS_SndData(void);
+ void RTS_SndData(const String &, unsigned long, unsigned char = VarAddr_W);
+ void RTS_SndData(const char[], unsigned long, unsigned char = VarAddr_W);
+ void RTS_SndData(char, unsigned long, unsigned char = VarAddr_W);
+ void RTS_SndData(unsigned char*, unsigned long, unsigned char = VarAddr_W);
+ void RTS_SndData(int, unsigned long, unsigned char = VarAddr_W);
+ void RTS_SndData(float, unsigned long, unsigned char = VarAddr_W);
+ void RTS_SndData(unsigned int,unsigned long, unsigned char = VarAddr_W);
+ void RTS_SndData(long,unsigned long, unsigned char = VarAddr_W);
+ void RTS_SndData(unsigned long,unsigned long, unsigned char = VarAddr_W);
+ void RTS_SDcard_Stop();
+ void RTS_HandleData();
+ void RTS_Init();
+
+ DB recdat;
+ DB snddat;
+ unsigned char databuf[SizeofDatabuf];
+
+ static rx_datagram_state_t rx_datagram_state;
+ static uint8_t rx_datagram_len;
+ static bool Initialized;
+ };
+
+ static RTSSHOW rtscheck;
+
+ #define Addvalue 3
+ #define PrintChoice_Value (0+Addvalue)
+ #define Zoffset_Value (3+Addvalue)
+ #define Setting_Value (8+Addvalue)
+ #define XYZEaxis_Value (12+Addvalue)
+ #define Filament_Value (15+Addvalue)
+ #define Language_Value (18+Addvalue)
+ #define Filename_Value (22+Addvalue)
+
+ enum PROC_COM {
+ Printfile = 0,
+ Adjust,
+ Feedrate,
+ PrintChoice = PrintChoice_Value,
+ Zoffset = Zoffset_Value,
+ TempControl,
+ ManualSetTemp,
+ Setting = Setting_Value,
+ ReturnBack,
+ Bedlevel,
+ Autohome,
+ XYZEaxis = XYZEaxis_Value,
+ Filament = Filament_Value,
+ LanguageChoice = Language_Value,
+ No_Filament,
+ PwrOffNoF,
+ Volume,
+ Filename = Filename_Value
+ };
+
+ const unsigned long Addrbuf[] = {
+ 0x1002, 0x1004, 0x1006, 0x1008, 0x100A, 0x100C, 0x1026, 0x1030, 0x1032, 0x1034, 0x103A,
+ 0x103E, 0x1040, 0x1044, 0x1046, 0x1048, 0x104A, 0x104C, 0x1054, 0x1056, 0x1058,
+ 0x105C, 0x105E, 0x105F, 0x1088, 0
+ };
+
+ void RTSUpdate();
+ void RTSInit();
+
+} // ExtUI
+
+#ifndef MAIN_MENU_ITEM_1_GCODE
+ #if ENABLED(AUTO_BED_LEVELING_BILINEAR)
+ #define MEASURING_GCODE "M190S55\nG28O\nG34\nG29\nM400\nM104S215\nG28\nM109S215\nM420S1\nG1X100Y100F5000\nG1Z0\nM500\nM117 Set Z Offset"
+ #elif ENABLED(AUTO_BED_LEVELING_UBL)
+ #define MEASURING_GCODE "M190S55\nG28O\nG34\nG29P1\nG29P3\nG29S1\nG29S0\nG29F0.0\nG29A\nM104S215\nG28\nM109S215\nG1X150Y150F5000\nG1Z0\nM500\nM400\nM117 Set Z Offset"
+ #else
+ #define MEASURING_GCODE "G28"
+ #endif
+#else
+ #define MEASURING_GCODE MAIN_MENU_ITEM_1_GCODE
+#endif
diff --git a/Marlin/src/lcd/extui/mks_ui/tft_multi_language.cpp b/Marlin/src/lcd/extui/mks_ui/tft_multi_language.cpp
index 6e9c37b0169e..aaddbcb892d2 100644
--- a/Marlin/src/lcd/extui/mks_ui/tft_multi_language.cpp
+++ b/Marlin/src/lcd/extui/mks_ui/tft_multi_language.cpp
@@ -167,7 +167,7 @@ void machine_setting_disp() {
machine_menu.Extrude_Min_Temper = EXTRUD_MIN_TEMPER_CN;
machine_menu.HotbedConfTitle = HOTBED_CONF_TITLE_CN;
- machine_menu.HotbedAjustType = HOTBED_ADJUST_CN;
+ machine_menu.HotbedAdjustType = HOTBED_ADJUST_CN;
machine_menu.HotbedMinTemperature = HOTBED_MIN_TEMPERATURE_CN;
machine_menu.HotbedMaxTemperature = HOTBED_MAX_TEMPERATURE_CN;
@@ -395,7 +395,7 @@ void machine_setting_disp() {
machine_menu.Extrude_Min_Temper = EXTRUD_MIN_TEMPER_T_CN;
machine_menu.HotbedConfTitle = HOTBED_CONF_TITLE_T_CN;
- machine_menu.HotbedAjustType = HOTBED_ADJUST_T_CN;
+ machine_menu.HotbedAdjustType = HOTBED_ADJUST_T_CN;
machine_menu.HotbedMinTemperature = HOTBED_MIN_TEMPERATURE_T_CN;
machine_menu.HotbedMaxTemperature = HOTBED_MAX_TEMPERATURE_T_CN;
@@ -627,7 +627,7 @@ void machine_setting_disp() {
machine_menu.HotbedEnable = HOTBED_ENABLE_EN;
machine_menu.HotbedConfTitle = HOTBED_CONF_TITLE_EN;
- machine_menu.HotbedAjustType = HOTBED_ADJUST_EN;
+ machine_menu.HotbedAdjustType = HOTBED_ADJUST_EN;
machine_menu.HotbedMinTemperature = HOTBED_MIN_TEMPERATURE_EN;
machine_menu.HotbedMaxTemperature = HOTBED_MAX_TEMPERATURE_EN;
diff --git a/Marlin/src/lcd/extui/mks_ui/tft_multi_language.h b/Marlin/src/lcd/extui/mks_ui/tft_multi_language.h
index 2a5135dded4f..9a6109cbe7c3 100644
--- a/Marlin/src/lcd/extui/mks_ui/tft_multi_language.h
+++ b/Marlin/src/lcd/extui/mks_ui/tft_multi_language.h
@@ -150,7 +150,7 @@ typedef struct machine_common_disp {
const char *HotbedEnable;
const char *HotbedConfTitle;
- const char *HotbedAjustType;
+ const char *HotbedAdjustType;
const char *HotbedMinTemperature;
const char *HotbedMaxTemperature;
diff --git a/buildroot/tests/FYSETC_F6 b/buildroot/tests/FYSETC_F6
index 91b1f899ffea..24fe550157ad 100755
--- a/buildroot/tests/FYSETC_F6
+++ b/buildroot/tests/FYSETC_F6
@@ -10,8 +10,7 @@ set -e
# Build with the default configurations
#
restore_configs
-opt_set MOTHERBOARD BOARD_FYSETC_F6_13 LCD_SERIAL_PORT 1
-opt_enable DGUS_LCD_UI_FYSETC
+opt_set MOTHERBOARD BOARD_FYSETC_F6_13 LCD_SERIAL_PORT 1 DGUS_LCD_UI FYSETC
exec_test $1 $2 "FYSETC F6 1.3 with DGUS" "$3"
#
diff --git a/ini/features.ini b/ini/features.ini
index e376e2757e6e..c9c752865808 100644
--- a/ini/features.ini
+++ b/ini/features.ini
@@ -88,6 +88,7 @@ DGUS_LCD_UI_FYSETC = src_filter=+
DGUS_LCD_UI_HIPRECY = src_filter=+
DGUS_LCD_UI_MKS = src_filter=+
DGUS_LCD_UI_ORIGIN = src_filter=+
+DGUS_LCD_UI_IA_CREALITY = src_filter=+
EXTUI_EXAMPLE = src_filter=+
TOUCH_UI_FTDI_EVE = src_filter=+
MALYAN_LCD = src_filter=+
diff --git a/platformio.ini b/platformio.ini
index 09519532e79c..991c17b2ff7c 100644
--- a/platformio.ini
+++ b/platformio.ini
@@ -89,6 +89,7 @@ default_src_filter = + - - + -
-
-
+ -
-
-
- -