this is a little File Browser for the OdroidGo via the Webbrowser.
You can choose between Accespoint mode or connect to an existing Wifi:
The Web File Browser will create an Access Point with,
SSID: odroidgo
PSK: odroidgo
Webadress: http://192.168.4.1 (if you connected to the access point)
or
it will connect to the Wifi which was defined in the WIFI.TXT in the SD-Card root directory.
Webadress: http://displayedIPonOdroidGo or http://ordroidgo.local
WIFI.TXT
Code: Select all
MyWifiName
MyWifiPSK
You can create folders, delete and upload new files.
Github: https://github.com/ripper121/odroidgowebfilebrowser
Download: https://github.com/ripper121/odroidgowe ... Browser.fw
Code: Select all
#include <WiFi.h>
#include <WiFiClient.h>
#include <WebServer.h>
#include <ESPmDNS.h>
#include <odroid_go.h>
#include <WiFiAP.h>
#define LED_BUILTIN 2
const char* ssid = "odroidgo";
const char* password = "odroidgo";
const char* header = "<!DOCTYPE html>"
"<html>"
"<head>"
"<meta name=\"viewport\" content=\"width=device-width, initial-scale=1\">"
"<style>"
" h1 {"
" border-bottom: 1px solid #c0c0c0;"
" margin-bottom: 10px;"
" padding-bottom: 10px;"
" white-space: nowrap;"
" }"
" table {"
" border-collapse: collapse;"
" }"
" th {"
" cursor: pointer;"
" }"
" td.detailsColumn {"
" -webkit-padding-start: 2em;"
" text-align: end;"
" white-space: nowrap;"
" }"
" a.icon {"
" -webkit-padding-start: 1.5em;"
" text-decoration: none;"
" }"
" a.icon:hover {"
" text-decoration: underline;"
" }"
" a.file {"
" background : url(\" \") left top no-repeat;"
" }"
" a.dir {"
" background : url(\" \") left top no-repeat;"
" }"
" a.up {"
" background : url(\" \") left top no-repeat;"
" }"
" html[dir=rtl] a {"
" background-position-x: right;"
" }"
" #parentDirLinkBox {"
" margin-bottom: 10px;"
" padding-bottom: 10px;"
" }"
" #listingParsingErrorBox {"
" border: 1px solid black;"
" background: #fae691;"
" padding: 10px;"
" display: none;"
" }"
"</style>"
"<title id=\"title\">Web FileBrowser</title>"
"</head>"
"<body>";
const char* footer = "<hr>"
"<br>Made with ♥ and ☕ by ripper121<br>"
"<hr>"
"</body>"
"</html>";
const char* script = "<script>function sortTable(l){var e=document.getElementById(\"theader\"),n=e.cells[l].dataset.order||\"1\",s=0-(n=parseInt(n,10));e.cells[l].dataset.order=s;var t,a=document.getElementById(\"tbody\"),r=a.rows,d=[];for(t=0;t<r.length;t++)d.push(r[t]);for(d.sort(function(e,t){var a=e.cells[l].dataset.value,r=t.cells[l].dataset.value;return l?(a=parseInt(a,10),(r=parseInt(r,10))<a?s:a<r?n:0):r<a?s:a<r?n:0}),t=0;t<d.length;t++)a.appendChild(d[t])}</script>";
String uploadPath = "";
WebServer server(80);
String getContentType(String filename) {
filename.toUpperCase();
if (server.hasArg("download")) return "application/octet-stream";
else if (filename.endsWith(".HTM")) return "text/html";
else if (filename.endsWith(".HTML")) return "text/html";
else if (filename.endsWith(".CSS")) return "text/css";
else if (filename.endsWith(".JS")) return "application/javascript";
else if (filename.endsWith(".PNG")) return "image/png";
else if (filename.endsWith(".GIF")) return "image/gif";
else if (filename.endsWith(".JPG")) return "image/jpeg";
else if (filename.endsWith(".ICO")) return "image/x-icon";
else if (filename.endsWith(".XML")) return "text/xml";
else if (filename.endsWith(".PDF")) return "application/x-pdf";
else if (filename.endsWith(".ZIP")) return "application/x-zip";
else if (filename.endsWith(".GZ")) return "application/x-gzip";
return "text/plain";
}
bool handleFileRead(String path) {
Serial.println("handleFileRead: " + path);
if (path.endsWith("/")) path += "index.htm";
String contentType = getContentType(path);
String pathWithGz = path + ".gz";
if (SD.exists(pathWithGz) || SD.exists(path)) {
if (SD.exists(pathWithGz))
path += ".gz";
File file = SD.open(path, "r");
size_t sent = server.streamFile(file, contentType);
file.close();
return true;
}
return false;
}
void handleRoot() {
GO.lcd.clear();
GO.lcd.setCursor(0, 0);
String directory = urldecode(server.uri());
uploadPath = directory;
File dir = SD.open(directory);
GO.lcd.print("Index of ");GO.lcd.println(directory);
String entryName = "";
String tree = "";
bool emptyFolder = true;
while (true) {
File entry = dir.openNextFile();
entryName = entry.name();
entryName.replace(directory + "/", "");
if (! entry) {
// no more files
break;
}
if (entry.isDirectory()) {
tree += F("<tr>");
tree += F("<td data-value=\"");
tree += entryName;
tree += F("/\"><a class=\"icon dir\" href=\"");
tree += entry.name();
tree += F("\">");
tree += entryName;
tree += F("/</a></td>");
tree += F("<td class=\"detailsColumn\" data-value=\"0\">-</td>");
tree += F("<td class=\"detailsColumn\" data-value=\"0\">");
tree += F("<button class='buttons' onclick=\"location.href='/deleteConfirm?folder=");
tree += entry.name();
tree += F("';\">Delete</button></td>");
tree += F("</tr>");
GO.lcd.print(entryName); GO.lcd.println(" - <DIR>");
emptyFolder = false;
} else {
tree += F("<tr>");
tree += F("<td data-value=\"");
tree += entry.name();
tree += F("\"><a class=\"icon file\" draggable=\"true\" href=\"");
tree += entry.name();
tree += F("\">");
tree += entryName;
tree += F("</a></td>");
tree += F("<td class=\"detailsColumn\" data-value=\")");
tree += file_size(entry.size());
tree += F("\">");
tree += file_size(entry.size());
tree += F("</td>");
tree += F("<td class=\"detailsColumn\" data-value=\"0\">");
tree += F("<button class='buttons' onclick=\"location.href='/deleteConfirm?file=");
tree += entry.name();
tree += F("';\">Delete</button></td>");
tree += F("</tr>");
GO.lcd.print(entryName); GO.lcd.print(" - "); GO.lcd.println(file_size(entry.size()));
emptyFolder = false;
}
entry.close();
}
int i, count;
for (i = 0, count = 0; directory[i]; i++)
count += (directory[i] == '/');
count++;
int parserCnt = 0;
int rFromIndex = 0, rToIndex = -1;
String lastElement = "";
String tempElement = "";
String path = directory;
path.remove(0, 1);
path += "/";
while (count >= parserCnt) {
rFromIndex = rToIndex + 1;
rToIndex = path.indexOf('/', rFromIndex);
if (count == parserCnt) {
if (rToIndex == 0 || rToIndex == -1) break;
tempElement = lastElement;
lastElement = path.substring(rFromIndex, rToIndex);
} else parserCnt++;
}
/*
Serial.print("directory:");
Serial.println(directory);
Serial.print("path:");
Serial.println(path);
Serial.print("lastElement:");
Serial.println(lastElement);
*/
String webpage = "";
webpage += F(header);
webpage += F("<h1 id=\"header\">Index of ");
webpage += directory;
webpage += F("</h1>");
if (directory != "/") {
webpage += F("<div id=\"parentDirLinkBox\" style=\"display:block;\">");
webpage += F("<a id=\"parentDirLink\" class=\"icon up\" href=\"");
directory.replace(lastElement, "");
if (directory.length() > 1)
directory = directory.substring(0, directory.length() - 1);
webpage += directory;
webpage += F("\">");
webpage += F("<span id=\"parentDirText\">[Parent directory]</span>");
webpage += F("</a>");
webpage += F("</div>");
}
webpage += F(script);
webpage += F("<table>");
webpage += F("<thead>");
webpage += F("<tr class=\"header\" id=\"theader\">");
webpage += F("<th onclick=\"sortTable(0);\">Name</th>");
webpage += F("<th class=\"detailsColumn\" onclick=\"sortTable(1);\">Size</th>");
webpage += F("<th></th>");
webpage += F("</tr>");
webpage += F("</thead>");
webpage += F("<tbody id=\"tbody\">");
webpage += tree;
webpage += F("</tbody>");
webpage += F("</table>");
webpage += F("<hr>");
webpage += F("<FORM action='/fupload' method='post' enctype='multipart/form-data'>");
webpage += F("<input class='buttons' type='file' name='fupload' id = 'fupload' value=''>");
webpage += F("<button class='buttons' type='submit'>Upload</button></form><br>");
webpage += F("<FORM action='/mkdir' method='post' enctype='multipart/form-data'>");
webpage += F("<input type='hidden' id='path' name='path' value='");
webpage += uploadPath;
webpage += F("'>");
webpage += F("<input class='buttons' name='dirName' id ='dirName' value='NewFolder'>");
webpage += F("<button class='buttons' type='submit'>MkDir</button></form>");
webpage += F(footer);
if (tree == "") {
String dlPath = urldecode(server.uri());
if (SD.exists(dlPath)) {
File entry = SD.open(dlPath);
if (!entry.isDirectory()) {
Serial.println(dlPath);
handleFileRead(dlPath);
}
}
else {
handleNotFound();
}
}
server.send(200, "text/html", webpage);
}
void handleNotFound() {
String webpage = "";
webpage += F(header);
webpage += F("<hr>File Not Found<br>");
webpage += F("<br>URI:");
webpage += server.uri();
webpage += F("<br>Method: ");
webpage += (server.method() == HTTP_GET) ? "GET" : "POST";
webpage += F("<br>Arguments: ");
webpage += server.args();
webpage += F("<br>");
for (uint8_t i = 0; i < server.args(); i++) {
webpage += server.argName(i) + ": " + server.arg(i) + "<br>";
}
webpage += F("<br><button class='buttons' onclick=\"location.href='/';\">OK</button>");
webpage += F(footer);
server.send(404, "text/html", webpage);
}
void doMkdir() {
GO.lcd.clear();
GO.lcd.setCursor(0, 0);
String webpage = "";
webpage += F(header);
String path = "";
String dirName = "";
for (uint8_t i = 0; i < server.args(); i++) {
if (server.argName(i) == "dirName") {
Serial.printf("Dir Name: %s\n", server.arg(i));
dirName = server.arg(i);
}
if (server.argName(i) == "path") {
Serial.printf("Path: %s\n", server.arg(i));
path = server.arg(i);
}
}
if (dirName != "" && path != "") {
webpage += F("<hr>Creating Dir: <br>");
if (path == "/")
path = path + dirName;
else
path = path + "/" + dirName;
webpage += path;
Serial.printf("Creating Dir: %s\n", path);
GO.lcd.print("Creating Dir:"); GO.lcd.println(path);
if (SD.mkdir(path)) {
webpage += F("<br>Dir created<br>");
GO.lcd.println("Dir created");
} else {
Serial.println("mkdir failed");
webpage += F("<br>mkdir failed<br>");
GO.lcd.println("mkdir failed");
}
} else {
webpage += F("<br>Path or Name empty!");
GO.lcd.println("Path or Name empty!");
}
webpage += F("<br><button class='buttons' onclick=\"location.href='/';\">OK</button>");
webpage += F(footer);
server.send(200, "text/html", webpage);
}
void doDelete() {
GO.lcd.clear();
GO.lcd.setCursor(0, 0);
String webpage = "";
webpage += F(header);
for (uint8_t i = 0; i < server.args(); i++) {
if (server.argName(i) == "file") {
Serial.printf("Deleting file: %s\n", server.arg(i));
GO.lcd.print("Deleting file:"); GO.lcd.println(server.arg(i));
webpage += F("<hr>Deleting file: <br>");
webpage += server.arg(i);
if (SD.remove(server.arg(i))) {
webpage += F("<br>File deleted<br>");
GO.lcd.println("File deleted");
} else {
webpage += F("<br>Delete failed<br>");
GO.lcd.println("Delete failed");
}
}
if (server.argName(i) == "folder") {
Serial.printf("Removing Dir: %s\n", server.arg(i));
GO.lcd.print("Removing Dir:"); GO.lcd.println(server.arg(i));
webpage += F("<hr>Removing Dir: <br>");
webpage += server.arg(i);
if (SD.rmdir(server.arg(i))) {
webpage += F("<br>Dir removed<br>");
GO.lcd.println("Dir removed");
} else {
webpage += F("<br>rmdir failed<br>");
GO.lcd.println("rmdir failed");
}
}
}
webpage += F("<br><button class='buttons' onclick=\"location.href='/';\">OK</button>");
webpage += F(footer);
server.send(200, "text/html", webpage);
}
void deleteConfirm() {
String webpage = "";
webpage += F(header);
for (uint8_t i = 0; i < server.args(); i++) {
if (server.argName(i) == "file") {
webpage += F("<hr>Do you want to delete the file:<br>");
webpage += server.arg(i);
webpage += F("<br><br><button class='buttons' onclick=\"location.href='/doDelete?file=");
webpage += server.arg(i);
webpage += F("';\">Yes</button>");
}
if (server.argName(i) == "folder") {
webpage += F("<hr>Do you want to delete the Directory:<br>");
webpage += server.arg(i);
webpage += F("<br><br><button class='buttons' onclick=\"location.href='/doDelete?folder=");
webpage += server.arg(i);
webpage += F("';\">Yes</button>");
}
}
webpage += F("<button class='buttons' onclick='window.history.back();'>No</button>");
webpage += F(footer);
server.send(200, "text/html", webpage);
}
File UploadFile;
void handleFileUpload() {
// upload a new file to the Filing system
HTTPUpload& uploadfile = server.upload(); // See https://github.com/esp8266/Arduino/tree/master/libraries/ESP8266WebServer/srcv
// For further information on 'status' structure, there are other reasons such as a failed transfer that could be used
if (uploadfile.status == UPLOAD_FILE_START)
{
String filename = uploadfile.filename;
if (!filename.startsWith("/")) filename = uploadPath + "/" + filename;
GO.lcd.clear();
GO.lcd.setCursor(0, 0);
Serial.print("Upload File Name: "); Serial.println(filename);
GO.lcd.print("Upload File Name: "); GO.lcd.println(filename);
SD.remove(filename); // Remove a previous version, otherwise data is appended the file again
UploadFile = SD.open(filename, FILE_WRITE); // Open the file for writing in SPIFFS (create it, if doesn't exist)
filename = String();
}
else if (uploadfile.status == UPLOAD_FILE_WRITE)
{
if (UploadFile) {
UploadFile.write(uploadfile.buf, uploadfile.currentSize); // Write the received bytes to the file
GO.lcd.setCursor(0, 30);
GO.lcd.fillRect(0, 29, TFT_HEIGHT, 20, BLACK);
GO.lcd.print("Loading: "); GO.lcd.println(file_size(uploadfile.totalSize));
}
}
else if (uploadfile.status == UPLOAD_FILE_END)
{
if (UploadFile) // If the file was successfully created
{
UploadFile.close(); // Close the file again
delay(1000);
GO.lcd.clear();
GO.lcd.setCursor(0, 0);
Serial.print("Upload Size: "); Serial.println(uploadfile.totalSize);
GO.lcd.println("File was successfully uploaded.");
GO.lcd.print("Uploaded File Name: "); GO.lcd.println(uploadfile.filename);
GO.lcd.print("File Size: "); GO.lcd.println(file_size(uploadfile.totalSize));
String webpage = "";
webpage += F(header);
webpage += F("<hr>File was successfully uploaded<br>");
webpage += F("Uploaded File Name: ");
webpage += uploadfile.filename + "<br>";
webpage += F("File Size: ");
webpage += file_size(uploadfile.totalSize) + "<br>";
webpage += "<button class='buttons' onclick='window.history.back();'>OK</button>";
webpage += F(footer);
server.send(200, "text/html", webpage);
}
else
{
delay(1000);
GO.lcd.clear();
GO.lcd.setCursor(0, 0);
GO.lcd.println("Could Not Create Uploaded File (write-protected?)");
String webpage = "";
webpage += F(header);
webpage += F("<hr>Could Not Create Uploaded File (write-protected?)<br>");
webpage += "<button class='buttons' onclick='window.history.back();'>OK</button>";
webpage += F(footer);
server.send(200, "text/html", webpage);
}
}
}
String file_size(int bytes) {
String fsize = "";
if (bytes < 1024) fsize = String(bytes) + " B";
else if (bytes < (1024 * 1024)) fsize = String(bytes / 1024.0, 3) + " KB";
else if (bytes < (1024 * 1024 * 1024)) fsize = String(bytes / 1024.0 / 1024.0, 3) + " MB";
else fsize = String(bytes / 1024.0 / 1024.0 / 1024.0, 3) + " GB";
return fsize;
}
String urldecode(String str)
{
String encodedString = "";
char c;
char code0;
char code1;
for (int i = 0; i < str.length(); i++) {
c = str.charAt(i);
if (c == '+') {
encodedString += ' ';
} else if (c == '%') {
i++;
code0 = str.charAt(i);
i++;
code1 = str.charAt(i);
c = (h2int(code0) << 4) | h2int(code1);
encodedString += c;
} else {
encodedString += c;
}
yield();
}
return encodedString;
}
String urlencode(String str)
{
String encodedString = "";
char c;
char code0;
char code1;
char code2;
for (int i = 0; i < str.length(); i++) {
c = str.charAt(i);
if (c == ' ') {
encodedString += '+';
} else if (isalnum(c)) {
encodedString += c;
} else {
code1 = (c & 0xf) + '0';
if ((c & 0xf) > 9) {
code1 = (c & 0xf) - 10 + 'A';
}
c = (c >> 4) & 0xf;
code0 = c + '0';
if (c > 9) {
code0 = c - 10 + 'A';
}
code2 = '\0';
encodedString += '%';
encodedString += code0;
encodedString += code1;
//encodedString+=code2;
}
yield();
}
return encodedString;
}
unsigned char h2int(char c)
{
if (c >= '0' && c <= '9') {
return ((unsigned char)c - '0');
}
if (c >= 'a' && c <= 'f') {
return ((unsigned char)c - 'a' + 10);
}
if (c >= 'A' && c <= 'F') {
return ((unsigned char)c - 'A' + 10);
}
return (0);
}
String split(String s, char parser, int index) {
String rs = "";
int parserIndex = index;
int parserCnt = 0;
int rFromIndex = 0, rToIndex = -1;
while (index >= parserCnt) {
rFromIndex = rToIndex + 1;
rToIndex = s.indexOf(parser, rFromIndex);
if (index == parserCnt) {
if (s.substring(rFromIndex, rToIndex) == "")
break;
} else parserCnt++;
rs = s.substring(rFromIndex, rToIndex);
}
return rs;
}
void setup(void) {
GO.begin();
GO.Speaker.setVolume(0);
pinMode(25, OUTPUT);
digitalWrite(25, LOW);
GO.battery.setProtection(true);
delay(100);
GO.lcd.clear();
GO.lcd.setCursor(0, 0);
GO.lcd.setTextWrap(false);
Serial.println("");
Serial.print("Initializing SD card...");
GO.lcd.println("Initializing SD card...");
if (!SD.begin()) {
GO.lcd.println("Card Mount Failed");
Serial.println("Card Mount Failed");
GO.update();
return;
}
uint8_t cardType = SD.cardType();
if (cardType == CARD_NONE) {
GO.lcd.println("No SD card attached");
Serial.println("No SD card attached");
GO.update();
return;
}
Serial.print("SD Card Type: ");
if (cardType == CARD_MMC) {
GO.lcd.println("MMC");
Serial.println("MMC");
} else if (cardType == CARD_SD) {
GO.lcd.println("SDSC");
Serial.println("SDSC");
} else if (cardType == CARD_SDHC) {
GO.lcd.println("SDHC");
Serial.println("SDHC");
} else {
GO.lcd.println("UNKNOWN");
Serial.println("UNKNOWN");
}
uint64_t cardSize = SD.cardSize() / (1024 * 1024);
Serial.printf("SD Card Size: %lluMB\n", cardSize);
GO.lcd.printf("SD Card Size: %lluMB\n", cardSize);
GO.lcd.println(" ");
Serial.println();
GO.lcd.println("-Wifi Mode-");
GO.lcd.println("-> Press A for Access Point mode");
GO.lcd.println("-> Press B to connect to a Wifi");
GO.lcd.println(" ");
Serial.println();
while (true) {
if (GO.BtnA.isPressed()) {
// You can remove the password parameter if you want the AP to be open.
WiFi.softAP(ssid, password);
IPAddress myIP = WiFi.softAPIP();
Serial.print("AP IP address: "); Serial.println(myIP);
GO.lcd.println("AP started");
GO.lcd.print("AP SSID: "); GO.lcd.println(ssid);
GO.lcd.print("AP Password: "); GO.lcd.println(password);
GO.lcd.print("AP IP address: "); GO.lcd.println(myIP);
break;
}
if (GO.BtnB.isPressed()) {
String WifiSSID = "";
String WifiPSK = "";
String path = "/WIFI.TXT";
Serial.print("Reading file: ");
Serial.println(path);
GO.lcd.print("Reading file: ");
GO.lcd.println(path);
File wifiFile = SD.open(path);
if (!wifiFile) {
Serial.println("Failed to open file for reading");
GO.lcd.println("Failed to open file for reading");
return;
}
Serial.println("Read from file: ");
GO.lcd.println("Read from file: ");
while (wifiFile.available()) {
WifiSSID = wifiFile.readStringUntil('\n');
WifiSSID.replace("\r", "");
WifiPSK = wifiFile.readStringUntil('\n');
WifiPSK.replace("\r", "");
break;
}
wifiFile.close();
Serial.print("SSID: ");
Serial.println(WifiSSID);
Serial.print("PSK: ");
Serial.println(WifiPSK);
GO.lcd.print("SSID: '");
GO.lcd.print(WifiSSID);
GO.lcd.println("'");
GO.lcd.print("PSK: '");
GO.lcd.print(WifiPSK);
GO.lcd.println("'");
Serial.println();
//delete old wifi Credentials
WiFi.disconnect();
WiFi.begin(WifiSSID.c_str(), WifiPSK.c_str());
Serial.print("Connecting Wifi");
GO.lcd.print("Connecting Wifi");
while (WiFi.status() != WL_CONNECTED) {
delay(500);
Serial.print(".");
GO.lcd.print(".");
}
GO.lcd.println(" ");
Serial.println();
Serial.println("IP address: ");
Serial.println(WiFi.localIP());
GO.lcd.println("IP address: ");
GO.lcd.println(WiFi.localIP());
GO.lcd.println(" ");
Serial.println();
if (MDNS.begin("odroidgo")) {
Serial.println("MDNS responder started");
GO.lcd.println("MDNS responder started");
Serial.println("You can connect via http://odroidgo.local");
GO.lcd.println("You can connect via http://odroidgo.local");
}
break;
}
GO.update();
}
GO.lcd.println(" ");
Serial.println();
server.on("/", []() {
handleRoot();
});
server.onNotFound(handleRoot);
server.on("/fupload", HTTP_POST, []() {
server.send(200);
}, handleFileUpload);
server.on("/deleteConfirm", deleteConfirm);
server.on("/doDelete", doDelete);
server.on("/mkdir", doMkdir);
server.begin();
Serial.println("HTTP server started");
GO.lcd.println("HTTP server started");
GO.lcd.println();
Serial.println();
Serial.println("Initialization done.");
GO.lcd.println("Initializing done.");
}
void loop(void) {
GO.update();
server.handleClient();
}