/** * Marlin 3D Printer Firmware * Copyright (c) 2020 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/anycubic_chiron/FileNavigator.cpp * * Extensible_UI implementation for Anycubic Chiron * Written By Nick Wells, 2020 [https://github.com/SwiftNick] * (not affiliated with Anycubic, Ltd.) * * The AC panel wants files in block of 4 and can only display a flat list * This library allows full folder traversal or flat file display and supports both standerd and new style panels. * * ## Old Style TFT panel * Supported chars {}[]-+=_"$%^&*()~<>| * Max display length 22 chars * Max path len 29 chars * (DOS 8.3 filepath max 29chars) * (long filepath Max 22) * * ## New TFT Panel Format file display format * Supported chars {}[]-+=_!"$%^&*()~<>\| * Max display length 26 chars * Max path len 29 chars * (DOS 8.3 filepath must end '.GCO') * (long filepath must end '.gcode') * */ #include "../../../inc/MarlinConfigPre.h" #if ENABLED(ANYCUBIC_LCD_CHIRON) #include "FileNavigator.h" #include "chiron_tft.h" using namespace ExtUI; #define DEBUG_OUT ACDEBUG(AC_FILE) #include "../../../core/debug_out.h" namespace Anycubic { FileNavigator filenavigator; FileList FileNavigator::filelist; // Instance of the Marlin file API uint16_t FileNavigator::lastpanelindex; uint16_t FileNavigator::currentindex; // override the panel request uint8_t FileNavigator::currentfolderdepth; uint16_t FileNavigator::currentfolderindex[MAX_FOLDER_DEPTH]; // track folder pos for iteration char FileNavigator::currentfoldername[MAX_PATH_LEN + 1]; // Current folder path FileNavigator::FileNavigator() { reset(); } void FileNavigator::reset() { DEBUG_ECHOLNPGM("reset()"); currentfoldername[0] = '\0'; currentfolderdepth = 0; currentindex = 0; lastpanelindex = 0; ZERO(currentfolderindex); // Start at root folder while (!filelist.isAtRootDir()) filelist.upDir(); refresh(); } void FileNavigator::refresh() { filelist.refresh(); } void FileNavigator::changeDIR(const char *folder) { if (currentfolderdepth >= MAX_FOLDER_DEPTH) return; // limit the folder depth DEBUG_ECHOLNPGM("FD:" , folderdepth, " FP:",currentindex, " currentfolder:", currentfoldername, " enter:", folder); currentfolderindex[currentfolderdepth] = currentindex; strcat(currentfoldername, folder); strcat(currentfoldername, "/"); filelist.changeDir(folder); currentfolderdepth++; currentindex = 0; } void FileNavigator::upDIR() { DEBUG_ECHOLNPGM("upDIR() from D:", currentfolderdepth, " N:", currentfoldername); if (!filelist.isAtRootDir()) { filelist.upDir(); currentfolderdepth--; currentindex = currentfolderindex[currentfolderdepth]; // restore last position in the folder filelist.seek(currentindex); // restore file information } // Remove the child folder from the stored path if (currentfolderdepth == 0) currentfoldername[0] = '\0'; else { char * const pos = strchr(currentfoldername, '/'); *(pos + 1) = '\0'; } } void FileNavigator::skiptofileindex(uint16_t skip) { if (skip == 0) return; while (skip > 0) { if (filelist.seek(currentindex)) { DEBUG_ECHOLNPGM("CI:", currentindex, " FD:", currentfolderdepth, " N:", skip, " ", filelist.longFilename()); if (!filelist.isDir()) { skip--; currentindex++; } else changeDIR(filelist.shortFilename()); } // valid file if (currentindex == filelist.count()) { if (currentfolderdepth > 0) { upDIR(); currentindex++; } else break; // end of root folder } // end of folder } // files needed // No more files available. } #if ENABLED(AC_SD_FOLDER_VIEW) // SD Folder navigation void FileNavigator::getFiles(uint16_t index, panel_type_t paneltype, uint8_t filesneeded) { if (index == 0) currentindex = 0; // Each time we change folder we reset the file index to 0 and keep track // of the current position, since the TFT panel isn't aware of folder trees. if (index > 0) { --currentindex; // go back a file to take account of the .. we added to the root. if (index > lastpanelindex) currentindex += filesneeded; else currentindex = currentindex < 4 ? 0 : currentindex - filesneeded; } lastpanelindex = index; DEBUG_ECHOLNPGM("index=", index, " currentindex=", currentindex); if (currentindex == 0 && currentfolderdepth > 0) { // Add a link to go up a folder // The new panel ignores entries that don't end in .GCO or .gcode so add and pad them. if (paneltype <= AC_panel_new) { TFTSer.println("<<.GCO"); Chiron.SendtoTFTLN(F(".. .gcode")); } else { TFTSer.println("<<"); TFTSer.println(".."); } filesneeded--; } for (uint16_t seek = currentindex; seek < currentindex + filesneeded; seek++) { if (filelist.seek(seek)) { sendFile(paneltype); DEBUG_ECHOLNPGM("-", seek, " '", filelist.longFilename(), "' '", currentfoldername, "", filelist.shortFilename(), "'"); } } } void FileNavigator::sendFile(panel_type_t paneltype) { if (filelist.isDir()) { // Add mandatory tags for new panel otherwise lines are ignored. if (paneltype <= AC_panel_new) { TFTSer.print(filelist.shortFilename()); TFTSer.println(".GCO"); TFTSer.print(filelist.shortFilename()); TFTSer.write('/'); // Make sure we fill all 29 chars of the display line to clear the text buffer otherwise the last line is still visible for (int8_t i = strlen(filelist.shortFilename()); i < 19; i++) TFTSer.write(' '); TFTSer.println(".gcode"); } else { TFTSer.println(filelist.shortFilename()); TFTSer.print(filelist.shortFilename()); TFTSer.write('/'); TFTSer.println(); } } else { // Not DIR TFTSer.write('/'); if (currentfolderdepth > 0) TFTSer.print(currentfoldername); TFTSer.println(filelist.shortFilename()); TFTSer.print(filelist.longFilename()); // Make sure we fill all 29 chars of the display line to clear the text buffer otherwise the last line is still visible if (paneltype == AC_panel_new) for (int8_t i = strlen(filelist.longFilename()); i < 26; i++) TFTSer.write(' '); TFTSer.println(); } } // AC_SD_FOLDER_VIEW #else // Flat file list void FileNavigator::getFiles(uint16_t index, panel_type_t paneltype, uint8_t filesneeded) { DEBUG_ECHOLNPGM("getFiles() I:", index," L:", lastpanelindex); // if we're searching backwards, jump back to start and search forward if (index < lastpanelindex) { reset(); skiptofileindex(index); } lastpanelindex = index; while (filesneeded > 0) { if (filelist.seek(currentindex)) { if (!filelist.isDir()) { sendFile(paneltype); filesneeded--; currentindex++; } else changeDIR(filelist.shortFilename()); } // valid file if (currentindex == filelist.count()) { if (currentfolderdepth > 0) { upDIR(); currentindex++; } else break; // end of root folder } // end of folder } // files needed // No more files available. } void FileNavigator::sendFile(panel_type_t paneltype) { TFTSer.write('/'); if (currentfolderdepth > 0) TFTSer.print(currentfoldername); TFTSer.println(filelist.shortFilename()); if (currentfolderdepth > 0) TFTSer.print(currentfoldername); TFTSer.println(filelist.longFilename()); DEBUG_ECHOLNPGM("/", currentfoldername, "", filelist.shortFilename(), " ", filelist.longFilename()); } #endif // Flat file list } // Anycubic namespace #endif // ANYCUBIC_LCD_CHIRON