Webserver improvements
This commit is contained in:
Submodule components/cpputils updated: 43682439fe...47417e77b7
Submodule components/espcpputils updated: 6274c103b4...234dbb23fb
Submodule components/esphttpdutils updated: 4ebd651a70...148e02b6ca
@ -23,7 +23,7 @@ set(headers
|
|||||||
actions/switchprofileaction.h
|
actions/switchprofileaction.h
|
||||||
actions/updateswapfrontbackaction.h
|
actions/updateswapfrontbackaction.h
|
||||||
bluetoothmode.h
|
bluetoothmode.h
|
||||||
bobby_ble.h
|
ble_bobby.h
|
||||||
bletexthelpers.h
|
bletexthelpers.h
|
||||||
bmsutils.h
|
bmsutils.h
|
||||||
changevaluedisplay_bluetoothmode.h
|
changevaluedisplay_bluetoothmode.h
|
||||||
@ -91,7 +91,6 @@ set(headers
|
|||||||
displays/statusdisplay.h
|
displays/statusdisplay.h
|
||||||
displays/updatedisplay.h
|
displays/updatedisplay.h
|
||||||
esp_websocket_client.h
|
esp_websocket_client.h
|
||||||
htmltag.h
|
|
||||||
icons/alert.h
|
icons/alert.h
|
||||||
icons/bluetooth.h
|
icons/bluetooth.h
|
||||||
icons/bms.h
|
icons/bms.h
|
||||||
@ -134,7 +133,6 @@ set(headers
|
|||||||
dpad5wire.h
|
dpad5wire.h
|
||||||
feedbackparser.h
|
feedbackparser.h
|
||||||
globals.h
|
globals.h
|
||||||
htmlutils.h
|
|
||||||
macros_bobbycar.h
|
macros_bobbycar.h
|
||||||
modeinterface.h
|
modeinterface.h
|
||||||
ota.h
|
ota.h
|
||||||
@ -150,6 +148,9 @@ set(headers
|
|||||||
unifiedmodelmode.h
|
unifiedmodelmode.h
|
||||||
utils.h
|
utils.h
|
||||||
webserver.h
|
webserver.h
|
||||||
|
webserver_displaycontrol.h
|
||||||
|
webserver_ota.h
|
||||||
|
webserver_settings.h
|
||||||
wifitexthelpers.h
|
wifitexthelpers.h
|
||||||
wifi_bobbycar.h
|
wifi_bobbycar.h
|
||||||
)
|
)
|
||||||
|
@ -2,7 +2,7 @@
|
|||||||
|
|
||||||
// local includes
|
// local includes
|
||||||
#include "textinterface.h"
|
#include "textinterface.h"
|
||||||
#include "bobby_ble.h"
|
#include "ble_bobby.h"
|
||||||
|
|
||||||
namespace {
|
namespace {
|
||||||
#ifdef FEATURE_BLE
|
#ifdef FEATURE_BLE
|
||||||
|
@ -1,46 +0,0 @@
|
|||||||
#pragma once
|
|
||||||
|
|
||||||
#include <ESPAsyncWebServer.h>
|
|
||||||
|
|
||||||
namespace {
|
|
||||||
class HtmlTag {
|
|
||||||
public:
|
|
||||||
HtmlTag(AsyncResponseStream &stream, const char *tag);
|
|
||||||
|
|
||||||
template<typename T>
|
|
||||||
HtmlTag(AsyncResponseStream &stream, const char *tag, const T &x);
|
|
||||||
|
|
||||||
~HtmlTag();
|
|
||||||
|
|
||||||
private:
|
|
||||||
AsyncResponseStream &stream;
|
|
||||||
const char * const tag;
|
|
||||||
};
|
|
||||||
|
|
||||||
HtmlTag::HtmlTag(AsyncResponseStream &stream, const char *tag) :
|
|
||||||
stream{stream},
|
|
||||||
tag{tag}
|
|
||||||
{
|
|
||||||
stream.print("<");
|
|
||||||
stream.print(tag);
|
|
||||||
stream.print(">");
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename T>
|
|
||||||
HtmlTag::HtmlTag(AsyncResponseStream &stream, const char *tag, const T &x) :
|
|
||||||
stream{stream},
|
|
||||||
tag{tag}
|
|
||||||
{
|
|
||||||
stream.print("<");
|
|
||||||
stream.print(tag);
|
|
||||||
stream.print(x);
|
|
||||||
stream.print(">");
|
|
||||||
}
|
|
||||||
|
|
||||||
HtmlTag::~HtmlTag()
|
|
||||||
{
|
|
||||||
stream.print("</");
|
|
||||||
stream.print(tag);
|
|
||||||
stream.print(">");
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,91 +0,0 @@
|
|||||||
#pragma once
|
|
||||||
|
|
||||||
#include "htmltag.h"
|
|
||||||
|
|
||||||
namespace {
|
|
||||||
void breakLine(AsyncResponseStream &stream)
|
|
||||||
{
|
|
||||||
stream.print("<br/>");
|
|
||||||
}
|
|
||||||
|
|
||||||
void label(AsyncResponseStream &stream, const char *name, const char *text)
|
|
||||||
{
|
|
||||||
HtmlTag label(stream, "label", std::string(" for=\"") + name + "\"");
|
|
||||||
stream.print(text);
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename T>
|
|
||||||
void input(AsyncResponseStream &stream, T value, const char *type, const char *id, const char *name, const char *additionalAttributes = nullptr)
|
|
||||||
{
|
|
||||||
stream.print("<input type=\"");
|
|
||||||
stream.print(type);
|
|
||||||
if (id)
|
|
||||||
{
|
|
||||||
stream.print("\" id=\"");
|
|
||||||
stream.print(id);
|
|
||||||
}
|
|
||||||
stream.print("\" name=\"");
|
|
||||||
stream.print(name);
|
|
||||||
stream.print("\" value=\"");
|
|
||||||
stream.print(value);
|
|
||||||
stream.print("\"");
|
|
||||||
if (additionalAttributes)
|
|
||||||
stream.print(additionalAttributes);
|
|
||||||
stream.print("/>");
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename T>
|
|
||||||
void hiddenInput(AsyncResponseStream &stream, T value, const char *name)
|
|
||||||
{
|
|
||||||
input(stream, value, "hidden", nullptr, name);
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename T>
|
|
||||||
void numberInput(AsyncResponseStream &stream, T value, const char *id, const char *name, const char *text)
|
|
||||||
{
|
|
||||||
label(stream, id, text);
|
|
||||||
|
|
||||||
breakLine(stream);
|
|
||||||
|
|
||||||
input(stream, value, "number", id, name, " required");
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename T>
|
|
||||||
void numberInput(AsyncResponseStream &stream, T value, const char *name, const char *text)
|
|
||||||
{
|
|
||||||
numberInput(stream, value, name, name, text);
|
|
||||||
}
|
|
||||||
|
|
||||||
void submitButton(AsyncResponseStream &stream)
|
|
||||||
{
|
|
||||||
HtmlTag button(stream, "button", " type=\"submit\"");
|
|
||||||
stream.print("Submit");
|
|
||||||
}
|
|
||||||
|
|
||||||
void checkboxInput(AsyncResponseStream &stream, bool value, const char *id, const char *name, const char *text)
|
|
||||||
{
|
|
||||||
label(stream, id, text);
|
|
||||||
|
|
||||||
breakLine(stream);
|
|
||||||
|
|
||||||
input(stream, "on", "checkbox", id, name, value?" checked":"");
|
|
||||||
}
|
|
||||||
|
|
||||||
void checkboxInput(AsyncResponseStream &stream, bool value, const char *name, const char *text)
|
|
||||||
{
|
|
||||||
checkboxInput(stream, value, name, name, text);
|
|
||||||
}
|
|
||||||
|
|
||||||
void selectOption(AsyncResponseStream &stream, const char *value, const char *text, bool selected)
|
|
||||||
{
|
|
||||||
std::string str{" value=\""};
|
|
||||||
str += value;
|
|
||||||
str += "\"";
|
|
||||||
|
|
||||||
if (selected)
|
|
||||||
str += " selected";
|
|
||||||
|
|
||||||
HtmlTag option(stream, "option", str);
|
|
||||||
stream.print(text);
|
|
||||||
}
|
|
||||||
}
|
|
@ -131,7 +131,7 @@ using namespace std::chrono_literals;
|
|||||||
#endif
|
#endif
|
||||||
#endif
|
#endif
|
||||||
#ifdef FEATURE_BLE
|
#ifdef FEATURE_BLE
|
||||||
#include "bobby_ble.h"
|
#include "ble_bobby.h"
|
||||||
#endif
|
#endif
|
||||||
#ifdef FEATURE_WEBSERVER
|
#ifdef FEATURE_WEBSERVER
|
||||||
#include "webserver.h"
|
#include "webserver.h"
|
||||||
|
471
main/webserver.h
471
main/webserver.h
@ -11,85 +11,28 @@
|
|||||||
#include <esp_log.h>
|
#include <esp_log.h>
|
||||||
|
|
||||||
// 3rdparty lib includes
|
// 3rdparty lib includes
|
||||||
#include <fmt/core.h>
|
|
||||||
#include <espcppmacros.h>
|
#include <espcppmacros.h>
|
||||||
#include <espstrutils.h>
|
|
||||||
#include <strutils.h>
|
|
||||||
#include <numberparsing.h>
|
|
||||||
#include <esphttpdutils.h>
|
#include <esphttpdutils.h>
|
||||||
|
|
||||||
// local includes
|
// local includes
|
||||||
#include "screens.h"
|
#include "webserver_displaycontrol.h"
|
||||||
#include "textinterface.h"
|
|
||||||
#include "menudisplay.h"
|
|
||||||
#include "changevaluedisplay.h"
|
|
||||||
#include "displays/updatedisplay.h"
|
|
||||||
//#include "esputils.h"
|
|
||||||
#include "buttons.h"
|
|
||||||
#include "esphttpdutils.h"
|
|
||||||
#ifdef FEATURE_OTA
|
#ifdef FEATURE_OTA
|
||||||
#include "ota.h"
|
#include "webserver_ota.h"
|
||||||
#endif
|
#endif
|
||||||
#include "globals.h"
|
#include "webserver_settings.h"
|
||||||
|
|
||||||
namespace {
|
|
||||||
#ifdef FEATURE_WEBSERVER
|
#ifdef FEATURE_WEBSERVER
|
||||||
|
namespace {
|
||||||
httpd_handle_t httpdHandle;
|
httpd_handle_t httpdHandle;
|
||||||
|
|
||||||
std::atomic<bool> shouldReboot;
|
std::atomic<bool> shouldReboot;
|
||||||
|
|
||||||
class HtmlTag {
|
void initWebserver();
|
||||||
public:
|
void handleWebserver();
|
||||||
HtmlTag(std::string_view tagName, std::string &body) :
|
|
||||||
m_tagName{tagName},
|
|
||||||
m_body{body}
|
|
||||||
{
|
|
||||||
m_body += '<';
|
|
||||||
m_body += m_tagName;
|
|
||||||
m_body += '>';
|
|
||||||
}
|
|
||||||
|
|
||||||
HtmlTag(std::string_view tagName, std::string_view attributes, std::string &body) :
|
|
||||||
m_tagName{tagName},
|
|
||||||
m_body{body}
|
|
||||||
{
|
|
||||||
m_body += '<';
|
|
||||||
m_body += m_tagName;
|
|
||||||
if (!attributes.empty())
|
|
||||||
{
|
|
||||||
m_body += ' ';
|
|
||||||
m_body += attributes;
|
|
||||||
}
|
|
||||||
m_body += '>';
|
|
||||||
}
|
|
||||||
|
|
||||||
~HtmlTag()
|
|
||||||
{
|
|
||||||
m_body += "</";
|
|
||||||
m_body += m_tagName;
|
|
||||||
m_body += '>';
|
|
||||||
}
|
|
||||||
|
|
||||||
private:
|
|
||||||
const std::string_view m_tagName;
|
|
||||||
std::string &m_body;
|
|
||||||
};
|
|
||||||
|
|
||||||
esp_err_t webserver_root_handler(httpd_req_t *req);
|
|
||||||
esp_err_t webserver_up_handler(httpd_req_t *req);
|
|
||||||
esp_err_t webserver_down_handler(httpd_req_t *req);
|
|
||||||
esp_err_t webserver_confirm_handler(httpd_req_t *req);
|
|
||||||
esp_err_t webserver_back_handler(httpd_req_t *req);
|
|
||||||
esp_err_t webserver_triggerItem_handler(httpd_req_t *req);
|
|
||||||
esp_err_t webserver_setValue_handler(httpd_req_t *req);
|
|
||||||
esp_err_t webserver_reboot_handler(httpd_req_t *req);
|
esp_err_t webserver_reboot_handler(httpd_req_t *req);
|
||||||
#ifdef FEATURE_OTA
|
}
|
||||||
esp_err_t webserver_ota_handler(httpd_req_t *req);
|
|
||||||
esp_err_t webserver_trigger_ota_handler(httpd_req_t *req);
|
|
||||||
#endif
|
|
||||||
esp_err_t webserver_settings_handler(httpd_req_t *req);
|
|
||||||
esp_err_t webserver_save_settings_handler(httpd_req_t *req);
|
|
||||||
|
|
||||||
|
namespace {
|
||||||
void initWebserver()
|
void initWebserver()
|
||||||
{
|
{
|
||||||
shouldReboot = false;
|
shouldReboot = false;
|
||||||
@ -106,20 +49,17 @@ void initWebserver()
|
|||||||
}
|
}
|
||||||
|
|
||||||
for (const httpd_uri_t &uri : {
|
for (const httpd_uri_t &uri : {
|
||||||
httpd_uri_t { .uri = "/", .method = HTTP_GET, .handler = webserver_root_handler, .user_ctx = NULL },
|
httpd_uri_t { .uri = "/", .method = HTTP_GET, .handler = webserver_root_handler, .user_ctx = NULL },
|
||||||
httpd_uri_t { .uri = "/up", .method = HTTP_GET, .handler = webserver_up_handler, .user_ctx = NULL },
|
httpd_uri_t { .uri = "/triggerButton", .method = HTTP_GET, .handler = webserver_triggerButton_handler, .user_ctx = NULL },
|
||||||
httpd_uri_t { .uri = "/down", .method = HTTP_GET, .handler = webserver_down_handler, .user_ctx = NULL },
|
httpd_uri_t { .uri = "/triggerItem", .method = HTTP_GET, .handler = webserver_triggerItem_handler, .user_ctx = NULL },
|
||||||
httpd_uri_t { .uri = "/confirm", .method = HTTP_GET, .handler = webserver_confirm_handler, .user_ctx = NULL },
|
httpd_uri_t { .uri = "/setValue", .method = HTTP_GET, .handler = webserver_setValue_handler, .user_ctx = NULL },
|
||||||
httpd_uri_t { .uri = "/back", .method = HTTP_GET, .handler = webserver_back_handler, .user_ctx = NULL },
|
httpd_uri_t { .uri = "/reboot", .method = HTTP_GET, .handler = webserver_reboot_handler, .user_ctx = NULL },
|
||||||
httpd_uri_t { .uri = "/triggerItem", .method = HTTP_GET, .handler = webserver_triggerItem_handler, .user_ctx = NULL },
|
|
||||||
httpd_uri_t { .uri = "/setValue", .method = HTTP_GET, .handler = webserver_setValue_handler, .user_ctx = NULL },
|
|
||||||
httpd_uri_t { .uri = "/reboot", .method = HTTP_GET, .handler = webserver_reboot_handler, .user_ctx = NULL },
|
|
||||||
#ifdef FEATURE_OTA
|
#ifdef FEATURE_OTA
|
||||||
httpd_uri_t { .uri = "/ota", .method = HTTP_GET, .handler = webserver_ota_handler, .user_ctx = NULL },
|
httpd_uri_t { .uri = "/ota", .method = HTTP_GET, .handler = webserver_ota_handler, .user_ctx = NULL },
|
||||||
httpd_uri_t { .uri = "/triggerOta", .method = HTTP_GET, .handler = webserver_trigger_ota_handler, .user_ctx = NULL },
|
httpd_uri_t { .uri = "/triggerOta", .method = HTTP_GET, .handler = webserver_trigger_ota_handler, .user_ctx = NULL },
|
||||||
#endif
|
#endif
|
||||||
httpd_uri_t { .uri = "/settings", .method = HTTP_GET, .handler = webserver_settings_handler, .user_ctx = NULL },
|
httpd_uri_t { .uri = "/settings", .method = HTTP_GET, .handler = webserver_settings_handler, .user_ctx = NULL },
|
||||||
httpd_uri_t { .uri = "/saveSettings", .method = HTTP_GET, .handler = webserver_save_settings_handler, .user_ctx = NULL },
|
httpd_uri_t { .uri = "/saveSettings", .method = HTTP_GET, .handler = webserver_save_settings_handler, .user_ctx = NULL },
|
||||||
})
|
})
|
||||||
{
|
{
|
||||||
const auto result = httpd_register_uri_handler(httpdHandle, &uri);
|
const auto result = httpd_register_uri_handler(httpdHandle, &uri);
|
||||||
@ -138,387 +78,12 @@ void handleWebserver()
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
esp_err_t webserver_root_handler(httpd_req_t *req)
|
|
||||||
{
|
|
||||||
std::string body;
|
|
||||||
|
|
||||||
{
|
|
||||||
HtmlTag htmlTag{"html", body};
|
|
||||||
|
|
||||||
{
|
|
||||||
HtmlTag headTag{"head", body};
|
|
||||||
|
|
||||||
{
|
|
||||||
HtmlTag titleTag{"title", body};
|
|
||||||
body += "Bobbycar remote";
|
|
||||||
}
|
|
||||||
|
|
||||||
body += "<meta name=\"viewport\" content=\"width=device-width, initial-scale=1, shrink-to-fit=no\" />";
|
|
||||||
}
|
|
||||||
|
|
||||||
{
|
|
||||||
HtmlTag bodyTag{"body", body};
|
|
||||||
|
|
||||||
{
|
|
||||||
HtmlTag h1Tag{"h1", body};
|
|
||||||
body += "Bobbycar remote";
|
|
||||||
}
|
|
||||||
|
|
||||||
{
|
|
||||||
HtmlTag pTag{"p", body};
|
|
||||||
body += "<a href=\"/up\">Up</a> "
|
|
||||||
"<a href=\"/down\">Down</a> "
|
|
||||||
"<a href=\"/confirm\">Confirm</a> "
|
|
||||||
"<a href=\"/back\">Back</a> ";
|
|
||||||
|
|
||||||
#ifdef FEATURE_OTA
|
|
||||||
body += "<a href=\"/ota\">Update</a> ";
|
|
||||||
#endif
|
|
||||||
|
|
||||||
body += "<a href=\"/settings\">Settings</a>";
|
|
||||||
}
|
|
||||||
|
|
||||||
if (auto constCurrentDisplay = static_cast<const Display *>(currentDisplay.get()))
|
|
||||||
{
|
|
||||||
if (const auto *textInterface = constCurrentDisplay->asTextInterface())
|
|
||||||
{
|
|
||||||
HtmlTag h2Tag{"h2", body};
|
|
||||||
body += esphttpdutils::htmlentities(textInterface->text());
|
|
||||||
}
|
|
||||||
|
|
||||||
if (const auto *menuDisplay = constCurrentDisplay->asMenuDisplay())
|
|
||||||
{
|
|
||||||
HtmlTag ulTag{"ul", body};
|
|
||||||
|
|
||||||
int i{0};
|
|
||||||
menuDisplay->runForEveryMenuItem([&,selectedIndex=menuDisplay->selectedIndex()](const MenuItem &menuItem){
|
|
||||||
HtmlTag liTag = i == selectedIndex ?
|
|
||||||
HtmlTag{"li", "style=\"border: 1px solid black;\"", body} :
|
|
||||||
HtmlTag{"li", body};
|
|
||||||
|
|
||||||
body += fmt::format("<a href=\"/triggerItem?index={}\">{}</a>", i, esphttpdutils::htmlentities(menuItem.text()));
|
|
||||||
i++;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
else if (const auto *changeValueDisplay = constCurrentDisplay->asChangeValueDisplayInterface())
|
|
||||||
{
|
|
||||||
HtmlTag formTag{"form", "action=\"/setValue\" method=\"GET\"", body};
|
|
||||||
body += fmt::format("<input type=\"number\" name=\"value\" value=\"{}\" />", changeValueDisplay->shownValue());
|
|
||||||
body += "<button type=\"submit\">Update</button>";
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
body += "No web control implemented for current display.";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
body += "Currently no screen instantiated.";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
CALL_AND_EXIT_ON_ERROR(httpd_resp_set_type, req, "text/html")
|
|
||||||
CALL_AND_EXIT(httpd_resp_send, req, body.data(), body.size())
|
|
||||||
}
|
|
||||||
|
|
||||||
esp_err_t webserver_up_handler(httpd_req_t *req)
|
|
||||||
{
|
|
||||||
InputDispatcher::rotate(-1);
|
|
||||||
|
|
||||||
CALL_AND_EXIT_ON_ERROR(httpd_resp_set_type, req, "text/html")
|
|
||||||
CALL_AND_EXIT_ON_ERROR(httpd_resp_set_status, req, "302 Moved Permanently")
|
|
||||||
CALL_AND_EXIT_ON_ERROR(httpd_resp_set_hdr, req, "Location", "/")
|
|
||||||
constexpr const std::string_view body{"Ok, continue at <a href=\"/\">/</a>"};
|
|
||||||
CALL_AND_EXIT(httpd_resp_send, req, body.data(), body.size())
|
|
||||||
}
|
|
||||||
|
|
||||||
esp_err_t webserver_down_handler(httpd_req_t *req)
|
|
||||||
{
|
|
||||||
InputDispatcher::rotate(1);
|
|
||||||
|
|
||||||
CALL_AND_EXIT_ON_ERROR(httpd_resp_set_type, req, "text/html")
|
|
||||||
CALL_AND_EXIT_ON_ERROR(httpd_resp_set_status, req, "302 Moved Permanently")
|
|
||||||
CALL_AND_EXIT_ON_ERROR(httpd_resp_set_hdr, req, "Location", "/")
|
|
||||||
constexpr const std::string_view body{"Ok, continue at <a href=\"/\">/</a>"};
|
|
||||||
CALL_AND_EXIT(httpd_resp_send, req, body.data(), body.size())
|
|
||||||
}
|
|
||||||
|
|
||||||
esp_err_t webserver_confirm_handler(httpd_req_t *req)
|
|
||||||
{
|
|
||||||
InputDispatcher::confirmButton(true);
|
|
||||||
InputDispatcher::confirmButton(false);
|
|
||||||
|
|
||||||
CALL_AND_EXIT_ON_ERROR(httpd_resp_set_type, req, "text/html")
|
|
||||||
CALL_AND_EXIT_ON_ERROR(httpd_resp_set_status, req, "302 Moved Permanently")
|
|
||||||
CALL_AND_EXIT_ON_ERROR(httpd_resp_set_hdr, req, "Location", "/")
|
|
||||||
constexpr const std::string_view body{"Ok, continue at <a href=\"/\">/</a>"};
|
|
||||||
CALL_AND_EXIT(httpd_resp_send, req, body.data(), body.size())
|
|
||||||
}
|
|
||||||
|
|
||||||
esp_err_t webserver_back_handler(httpd_req_t *req)
|
|
||||||
{
|
|
||||||
InputDispatcher::backButton(true);
|
|
||||||
InputDispatcher::backButton(false);
|
|
||||||
|
|
||||||
CALL_AND_EXIT_ON_ERROR(httpd_resp_set_type, req, "text/html")
|
|
||||||
CALL_AND_EXIT_ON_ERROR(httpd_resp_set_status, req, "302 Moved Permanently")
|
|
||||||
CALL_AND_EXIT_ON_ERROR(httpd_resp_set_hdr, req, "Location", "/")
|
|
||||||
constexpr const std::string_view body{"Ok, continue at <a href=\"/\">/</a>"};
|
|
||||||
CALL_AND_EXIT(httpd_resp_send, req, body.data(), body.size())
|
|
||||||
}
|
|
||||||
|
|
||||||
esp_err_t webserver_triggerItem_handler(httpd_req_t *req)
|
|
||||||
{
|
|
||||||
std::string query;
|
|
||||||
|
|
||||||
if (const size_t queryLength = httpd_req_get_url_query_len(req))
|
|
||||||
{
|
|
||||||
query.resize(queryLength);
|
|
||||||
CALL_AND_EXIT_ON_ERROR(httpd_req_get_url_query_str, req, query.data(), query.size() + 1)
|
|
||||||
}
|
|
||||||
|
|
||||||
std::size_t index;
|
|
||||||
|
|
||||||
constexpr const std::string_view indexParamName{"index"};
|
|
||||||
|
|
||||||
{
|
|
||||||
char valueBufEncoded[256];
|
|
||||||
if (const auto result = httpd_query_key_value(query.data(), indexParamName.data(), valueBufEncoded, 256); result == ESP_OK)
|
|
||||||
{
|
|
||||||
char valueBuf[257];
|
|
||||||
esphttpdutils::urldecode(valueBuf, valueBufEncoded);
|
|
||||||
|
|
||||||
std::string_view value{valueBuf};
|
|
||||||
|
|
||||||
if (auto parsed = cpputils::fromString<typeof(index)>(value))
|
|
||||||
{
|
|
||||||
index = *parsed;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
CALL_AND_EXIT(httpd_resp_send_err, req, HTTPD_400_BAD_REQUEST, fmt::format("could not parse {} {}", indexParamName, value).c_str());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else if (result != ESP_ERR_NOT_FOUND)
|
|
||||||
{
|
|
||||||
ESP_LOGE(TAG, "httpd_query_key_value() %.*s failed with %s", indexParamName.size(), indexParamName.data(), esp_err_to_name(result));
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
CALL_AND_EXIT(httpd_resp_send_err, req, HTTPD_400_BAD_REQUEST, fmt::format("{} not set", indexParamName).c_str());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!currentDisplay)
|
|
||||||
CALL_AND_EXIT(httpd_resp_send_err, req, HTTPD_400_BAD_REQUEST, "currentDisplay is null");
|
|
||||||
|
|
||||||
auto *menuDisplay = currentDisplay->asMenuDisplay();
|
|
||||||
if (!menuDisplay)
|
|
||||||
CALL_AND_EXIT(httpd_resp_send_err, req, HTTPD_400_BAD_REQUEST, "currentDisplay is not a menu display");
|
|
||||||
|
|
||||||
if (/*index < 0 ||*/ index >= menuDisplay->menuItemCount())
|
|
||||||
CALL_AND_EXIT(httpd_resp_send_err, req, HTTPD_400_BAD_REQUEST, fmt::format("{} out of range", indexParamName).c_str());
|
|
||||||
|
|
||||||
menuDisplay->getMenuItem(index).triggered();
|
|
||||||
|
|
||||||
CALL_AND_EXIT_ON_ERROR(httpd_resp_set_type, req, "text/html")
|
|
||||||
CALL_AND_EXIT_ON_ERROR(httpd_resp_set_status, req, "302 Moved Permanently")
|
|
||||||
CALL_AND_EXIT_ON_ERROR(httpd_resp_set_hdr, req, "Location", "/")
|
|
||||||
constexpr const std::string_view body{"Ok, continue at <a href=\"/\">/</a>"};
|
|
||||||
CALL_AND_EXIT(httpd_resp_send, req, body.data(), body.size())
|
|
||||||
}
|
|
||||||
|
|
||||||
esp_err_t webserver_setValue_handler(httpd_req_t *req)
|
|
||||||
{
|
|
||||||
std::string query;
|
|
||||||
|
|
||||||
if (const size_t queryLength = httpd_req_get_url_query_len(req))
|
|
||||||
{
|
|
||||||
query.resize(queryLength);
|
|
||||||
CALL_AND_EXIT_ON_ERROR(httpd_req_get_url_query_str, req, query.data(), query.size() + 1)
|
|
||||||
}
|
|
||||||
|
|
||||||
int newValue;
|
|
||||||
|
|
||||||
constexpr const std::string_view valueParamName{"value"};
|
|
||||||
|
|
||||||
{
|
|
||||||
char valueBufEncoded[256];
|
|
||||||
if (const auto result = httpd_query_key_value(query.data(), valueParamName.data(), valueBufEncoded, 256); result == ESP_OK)
|
|
||||||
{
|
|
||||||
char valueBuf[257];
|
|
||||||
esphttpdutils::urldecode(valueBuf, valueBufEncoded);
|
|
||||||
|
|
||||||
std::string_view value{valueBuf};
|
|
||||||
|
|
||||||
if (auto parsed = cpputils::fromString<typeof(newValue)>(value))
|
|
||||||
{
|
|
||||||
newValue = *parsed;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
CALL_AND_EXIT(httpd_resp_send_err, req, HTTPD_400_BAD_REQUEST, fmt::format("could not parse {} {}", valueParamName, value).c_str());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else if (result != ESP_ERR_NOT_FOUND)
|
|
||||||
{
|
|
||||||
ESP_LOGE(TAG, "httpd_query_key_value() %.*s failed with %s", valueParamName.size(), valueParamName.data(), esp_err_to_name(result));
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
CALL_AND_EXIT(httpd_resp_send_err, req, HTTPD_400_BAD_REQUEST, fmt::format("{} not set", valueParamName).c_str());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!currentDisplay)
|
|
||||||
CALL_AND_EXIT(httpd_resp_send_err, req, HTTPD_400_BAD_REQUEST, "currentDisplay is null");
|
|
||||||
|
|
||||||
auto *changeValueDisplay = currentDisplay->asChangeValueDisplayInterface();
|
|
||||||
if (!changeValueDisplay)
|
|
||||||
CALL_AND_EXIT(httpd_resp_send_err, req, HTTPD_400_BAD_REQUEST, "currentDisplay is not a change value display");
|
|
||||||
|
|
||||||
changeValueDisplay->setShownValue(newValue);
|
|
||||||
|
|
||||||
CALL_AND_EXIT_ON_ERROR(httpd_resp_set_type, req, "text/html")
|
|
||||||
CALL_AND_EXIT_ON_ERROR(httpd_resp_set_status, req, "302 Moved Permanently")
|
|
||||||
CALL_AND_EXIT_ON_ERROR(httpd_resp_set_hdr, req, "Location", "/")
|
|
||||||
constexpr const std::string_view body{"Ok, continue at <a href=\"/\">/</a>"};
|
|
||||||
CALL_AND_EXIT(httpd_resp_send, req, body.data(), body.size())
|
|
||||||
}
|
|
||||||
|
|
||||||
esp_err_t webserver_reboot_handler(httpd_req_t *req)
|
esp_err_t webserver_reboot_handler(httpd_req_t *req)
|
||||||
{
|
{
|
||||||
shouldReboot = true;
|
shouldReboot = true;
|
||||||
|
|
||||||
CALL_AND_EXIT_ON_ERROR(httpd_resp_set_type, req, "text/html")
|
CALL_AND_EXIT(esphttpdutils::webserver_resp_send, req, esphttpdutils::ResponseStatus::Ok, "text/plain", "REBOOT called...")
|
||||||
std::string_view body{"REBOOT called..."};
|
|
||||||
CALL_AND_EXIT(httpd_resp_send, req, body.data(), body.size())
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#ifdef FEATURE_OTA
|
|
||||||
esp_err_t webserver_ota_handler(httpd_req_t *req)
|
|
||||||
{
|
|
||||||
std::string body;
|
|
||||||
|
|
||||||
{
|
|
||||||
HtmlTag htmlTag{"html", body};
|
|
||||||
|
|
||||||
{
|
|
||||||
HtmlTag headTag{"head", body};
|
|
||||||
|
|
||||||
{
|
|
||||||
HtmlTag titleTag{"title", body};
|
|
||||||
body += "Update";
|
|
||||||
}
|
|
||||||
|
|
||||||
body += "<meta name=\"viewport\" content=\"width=device-width, initial-scale=1, shrink-to-fit=no\" />";
|
|
||||||
}
|
|
||||||
|
|
||||||
{
|
|
||||||
HtmlTag bodyTag{"body", body};
|
|
||||||
|
|
||||||
{
|
|
||||||
HtmlTag h1Tag{"h1", body};
|
|
||||||
body += "Update";
|
|
||||||
}
|
|
||||||
|
|
||||||
{
|
|
||||||
HtmlTag pTag{"p", body};
|
|
||||||
body += "<a href=\"/\">Remote control</a> "
|
|
||||||
"<a href=\"/settings\">Settings</a>";
|
|
||||||
}
|
|
||||||
|
|
||||||
{
|
|
||||||
HtmlTag formTag{"form", "action=\"/triggerOta\" method=\"GET\"", body};
|
|
||||||
HtmlTag fieldsetTag{"fieldset", body};
|
|
||||||
{
|
|
||||||
HtmlTag legendTag{"legend", body};
|
|
||||||
body += "Trigger Update";
|
|
||||||
}
|
|
||||||
|
|
||||||
body += fmt::format("<input type=\"text\" name=\"url\" value=\"{}\" required />", esphttpdutils::htmlentities(stringSettings.otaUrl));
|
|
||||||
|
|
||||||
{
|
|
||||||
HtmlTag buttonTag{"button", "type=\"submit\"", body};
|
|
||||||
body += "Go";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
CALL_AND_EXIT_ON_ERROR(httpd_resp_set_type, req, "text/html")
|
|
||||||
CALL_AND_EXIT(httpd_resp_send, req, body.data(), body.size())
|
|
||||||
}
|
|
||||||
|
|
||||||
esp_err_t webserver_trigger_ota_handler(httpd_req_t *req)
|
|
||||||
{
|
|
||||||
return ESP_FAIL;
|
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
esp_err_t webserver_settings_handler(httpd_req_t *req)
|
|
||||||
{
|
|
||||||
std::string body;
|
|
||||||
|
|
||||||
{
|
|
||||||
HtmlTag htmlTag{"html", body};
|
|
||||||
|
|
||||||
{
|
|
||||||
HtmlTag headTag{"head", body};
|
|
||||||
|
|
||||||
{
|
|
||||||
HtmlTag titleTag{"title", body};
|
|
||||||
body += "Settings";
|
|
||||||
}
|
|
||||||
|
|
||||||
body += "<meta name=\"viewport\" content=\"width=device-width, initial-scale=1, shrink-to-fit=no\" />";
|
|
||||||
}
|
|
||||||
|
|
||||||
{
|
|
||||||
HtmlTag bodyTag{"body", body};
|
|
||||||
|
|
||||||
{
|
|
||||||
HtmlTag h1Tag{"h1", body};
|
|
||||||
body += "Settings";
|
|
||||||
}
|
|
||||||
|
|
||||||
{
|
|
||||||
HtmlTag pTag{"p", body};
|
|
||||||
body += "<a href=\"/\">Remote control</a> "
|
|
||||||
"<a href=\"/ota\">Update</a>";
|
|
||||||
}
|
|
||||||
|
|
||||||
stringSettings.executeForEveryCommonSetting([&](const char *key, auto value){
|
|
||||||
HtmlTag formTag{"form", "action=\"/saveSettings\" method=\"GET\"", body};
|
|
||||||
HtmlTag fieldsetTag{"fieldset", body};
|
|
||||||
{
|
|
||||||
HtmlTag legendTag{"legend", body};
|
|
||||||
body += esphttpdutils::htmlentities(key);
|
|
||||||
}
|
|
||||||
|
|
||||||
body += fmt::format("<input type=\"text\" name=\"{}\" value=\"{}\" required />",
|
|
||||||
esphttpdutils::htmlentities(key),
|
|
||||||
esphttpdutils::htmlentities(value));
|
|
||||||
|
|
||||||
{
|
|
||||||
HtmlTag buttonTag{"button", "type=\"submit\"", body};
|
|
||||||
body += "Save";
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
CALL_AND_EXIT_ON_ERROR(httpd_resp_set_type, req, "text/html")
|
|
||||||
CALL_AND_EXIT(httpd_resp_send, req, body.data(), body.size())
|
|
||||||
}
|
|
||||||
|
|
||||||
esp_err_t webserver_save_settings_handler(httpd_req_t *req)
|
|
||||||
{
|
|
||||||
return ESP_FAIL;
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
383
main/webserver_displaycontrol.h
Normal file
383
main/webserver_displaycontrol.h
Normal file
@ -0,0 +1,383 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
// esp-idf includes
|
||||||
|
#ifdef FEATURE_WEBSERVER
|
||||||
|
#include <esp_http_server.h>
|
||||||
|
#endif
|
||||||
|
#include <esp_log.h>
|
||||||
|
|
||||||
|
// 3rdparty lib includes
|
||||||
|
#include <htmlbuilder.h>
|
||||||
|
#include <fmt/core.h>
|
||||||
|
#include <espcppmacros.h>
|
||||||
|
#include <numberparsing.h>
|
||||||
|
#include <esphttpdutils.h>
|
||||||
|
#include <textinterface.h>
|
||||||
|
#include <menudisplay.h>
|
||||||
|
#include <changevaluedisplay.h>
|
||||||
|
|
||||||
|
// local includes
|
||||||
|
#include "buttons.h"
|
||||||
|
#include "globals.h"
|
||||||
|
|
||||||
|
#ifdef FEATURE_WEBSERVER
|
||||||
|
namespace {
|
||||||
|
esp_err_t webserver_root_handler(httpd_req_t *req);
|
||||||
|
esp_err_t webserver_triggerButton_handler(httpd_req_t *req);
|
||||||
|
esp_err_t webserver_triggerItem_handler(httpd_req_t *req);
|
||||||
|
esp_err_t webserver_setValue_handler(httpd_req_t *req);
|
||||||
|
} // namespace
|
||||||
|
|
||||||
|
using esphttpdutils::HtmlTag;
|
||||||
|
|
||||||
|
namespace {
|
||||||
|
esp_err_t webserver_root_handler(httpd_req_t *req)
|
||||||
|
{
|
||||||
|
std::string body;
|
||||||
|
|
||||||
|
{
|
||||||
|
HtmlTag htmlTag{"html", body};
|
||||||
|
|
||||||
|
{
|
||||||
|
HtmlTag headTag{"head", body};
|
||||||
|
|
||||||
|
{
|
||||||
|
HtmlTag titleTag{"title", body};
|
||||||
|
body += "Display control";
|
||||||
|
}
|
||||||
|
|
||||||
|
body += "<meta name=\"viewport\" content=\"width=device-width, initial-scale=1, shrink-to-fit=no\" />";
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
HtmlTag bodyTag{"body", body};
|
||||||
|
|
||||||
|
{
|
||||||
|
HtmlTag h1Tag{"h1", body};
|
||||||
|
body += "Display control";
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
HtmlTag pTag{"p", body};
|
||||||
|
body += "<b>Display control</b> "
|
||||||
|
#ifdef FEATURE_OTA
|
||||||
|
"<a href=\"/ota\">Update</a> "
|
||||||
|
#endif
|
||||||
|
|
||||||
|
"<a href=\"/settings\">Settings</a>";
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
HtmlTag pTag{"p", body};
|
||||||
|
body += "<a href=\"/triggerButton?button=up\">Up</a> "
|
||||||
|
"<a href=\"/triggerButton?button=down\">Down</a> "
|
||||||
|
"<a href=\"/triggerButton?button=confirm\">Confirm</a> "
|
||||||
|
"<a href=\"/triggerButton?button=back\">Back</a> "
|
||||||
|
"<a href=\"/triggerButton?button=profile0\">Profile0</a> "
|
||||||
|
"<a href=\"/triggerButton?button=profile1\">Profile1</a> "
|
||||||
|
"<a href=\"/triggerButton?button=profile2\">Profile2</a> "
|
||||||
|
"<a href=\"/triggerButton?button=profile3\">Profile3</a> ";
|
||||||
|
}
|
||||||
|
|
||||||
|
if (auto constCurrentDisplay = static_cast<const Display *>(currentDisplay.get()))
|
||||||
|
{
|
||||||
|
if (const auto *textInterface = constCurrentDisplay->asTextInterface())
|
||||||
|
{
|
||||||
|
HtmlTag h2Tag{"h2", body};
|
||||||
|
body += esphttpdutils::htmlentities(textInterface->text());
|
||||||
|
}
|
||||||
|
|
||||||
|
if (const auto *menuDisplay = constCurrentDisplay->asMenuDisplay())
|
||||||
|
{
|
||||||
|
HtmlTag ulTag{"ul", body};
|
||||||
|
|
||||||
|
int i{0};
|
||||||
|
menuDisplay->runForEveryMenuItem([&,selectedIndex=menuDisplay->selectedIndex()](const MenuItem &menuItem){
|
||||||
|
HtmlTag liTag = i == selectedIndex ?
|
||||||
|
HtmlTag{"li", "style=\"border: 1px solid black;\"", body} :
|
||||||
|
HtmlTag{"li", body};
|
||||||
|
|
||||||
|
body += fmt::format("<a href=\"/triggerItem?index={}\">{}</a>", i, esphttpdutils::htmlentities(menuItem.text()));
|
||||||
|
i++;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
else if (const auto *changeValueDisplay = constCurrentDisplay->asChangeValueDisplayInterface())
|
||||||
|
{
|
||||||
|
HtmlTag formTag{"form", "action=\"/setValue\" method=\"GET\"", body};
|
||||||
|
body += fmt::format("<input type=\"number\" name=\"value\" value=\"{}\" />", changeValueDisplay->shownValue());
|
||||||
|
body += "<button type=\"submit\">Update</button>";
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
body += "No web control implemented for current display.";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
body += "Currently no screen instantiated.";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
CALL_AND_EXIT(esphttpdutils::webserver_resp_send, req, esphttpdutils::ResponseStatus::Ok, "text/html", body)
|
||||||
|
}
|
||||||
|
|
||||||
|
esp_err_t webserver_triggerButton_handler(httpd_req_t *req)
|
||||||
|
{
|
||||||
|
std::string query;
|
||||||
|
if (auto result = esphttpdutils::webserver_get_query(req))
|
||||||
|
query = *result;
|
||||||
|
else
|
||||||
|
{
|
||||||
|
ESP_LOGE(TAG, "%.*s", result.error().size(), result.error().data());
|
||||||
|
CALL_AND_EXIT(esphttpdutils::webserver_resp_send, req, esphttpdutils::ResponseStatus::BadRequest, "text/plain", result.error());
|
||||||
|
}
|
||||||
|
|
||||||
|
std::string button;
|
||||||
|
constexpr const std::string_view buttonParamName{"button"};
|
||||||
|
|
||||||
|
{
|
||||||
|
char valueBufEncoded[256];
|
||||||
|
if (const auto result = httpd_query_key_value(query.data(), buttonParamName.data(), valueBufEncoded, 256); result != ESP_OK)
|
||||||
|
{
|
||||||
|
if (result == ESP_ERR_NOT_FOUND)
|
||||||
|
{
|
||||||
|
const auto msg = fmt::format("{} not set", buttonParamName);
|
||||||
|
ESP_LOGW(TAG, "%.*s", msg.size(), msg.data());
|
||||||
|
CALL_AND_EXIT(esphttpdutils::webserver_resp_send, req, esphttpdutils::ResponseStatus::BadRequest, "text/plain", msg);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
const auto msg = fmt::format("httpd_query_key_value() {} failed with {}", buttonParamName, esp_err_to_name(result));
|
||||||
|
ESP_LOGE(TAG, "%.*s", msg.size(), msg.data());
|
||||||
|
CALL_AND_EXIT(esphttpdutils::webserver_resp_send, req, esphttpdutils::ResponseStatus::BadRequest, "text/plain", msg);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
char valueBuf[257];
|
||||||
|
esphttpdutils::urldecode(valueBuf, valueBufEncoded);
|
||||||
|
|
||||||
|
button = valueBuf;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (button == "up")
|
||||||
|
{
|
||||||
|
InputDispatcher::rotate(-1);
|
||||||
|
|
||||||
|
CALL_AND_EXIT_ON_ERROR(httpd_resp_set_hdr, req, "Location", "/")
|
||||||
|
CALL_AND_EXIT(esphttpdutils::webserver_resp_send, req, esphttpdutils::ResponseStatus::TemporaryRedirect, "text/html", "Ok, continue at <a href=\"/\">/</a>")
|
||||||
|
}
|
||||||
|
else if (button == "down")
|
||||||
|
{
|
||||||
|
InputDispatcher::rotate(1);
|
||||||
|
|
||||||
|
CALL_AND_EXIT_ON_ERROR(httpd_resp_set_hdr, req, "Location", "/")
|
||||||
|
CALL_AND_EXIT(esphttpdutils::webserver_resp_send, req, esphttpdutils::ResponseStatus::TemporaryRedirect, "text/html", "Ok, continue at <a href=\"/\">/</a>")
|
||||||
|
}
|
||||||
|
else if (button == "confirm")
|
||||||
|
{
|
||||||
|
InputDispatcher::confirmButton(true);
|
||||||
|
InputDispatcher::confirmButton(false);
|
||||||
|
|
||||||
|
CALL_AND_EXIT_ON_ERROR(httpd_resp_set_hdr, req, "Location", "/")
|
||||||
|
CALL_AND_EXIT(esphttpdutils::webserver_resp_send, req, esphttpdutils::ResponseStatus::TemporaryRedirect, "text/html", "Ok, continue at <a href=\"/\">/</a>")
|
||||||
|
}
|
||||||
|
else if (button == "back")
|
||||||
|
{
|
||||||
|
InputDispatcher::backButton(true);
|
||||||
|
InputDispatcher::backButton(false);
|
||||||
|
|
||||||
|
CALL_AND_EXIT_ON_ERROR(httpd_resp_set_hdr, req, "Location", "/")
|
||||||
|
CALL_AND_EXIT(esphttpdutils::webserver_resp_send, req, esphttpdutils::ResponseStatus::TemporaryRedirect, "text/html", "Ok, continue at <a href=\"/\">/</a>")
|
||||||
|
}
|
||||||
|
else if (button == "profile0")
|
||||||
|
{
|
||||||
|
InputDispatcher::profileButton(0, true);
|
||||||
|
InputDispatcher::profileButton(0, false);
|
||||||
|
|
||||||
|
CALL_AND_EXIT_ON_ERROR(httpd_resp_set_hdr, req, "Location", "/")
|
||||||
|
CALL_AND_EXIT(esphttpdutils::webserver_resp_send, req, esphttpdutils::ResponseStatus::TemporaryRedirect, "text/html", "Ok, continue at <a href=\"/\">/</a>")
|
||||||
|
}
|
||||||
|
else if (button == "profile1")
|
||||||
|
{
|
||||||
|
InputDispatcher::profileButton(1, true);
|
||||||
|
InputDispatcher::profileButton(1, false);
|
||||||
|
|
||||||
|
CALL_AND_EXIT_ON_ERROR(httpd_resp_set_hdr, req, "Location", "/")
|
||||||
|
CALL_AND_EXIT(esphttpdutils::webserver_resp_send, req, esphttpdutils::ResponseStatus::TemporaryRedirect, "text/html", "Ok, continue at <a href=\"/\">/</a>")
|
||||||
|
}
|
||||||
|
else if (button == "profile2")
|
||||||
|
{
|
||||||
|
InputDispatcher::profileButton(2, true);
|
||||||
|
InputDispatcher::profileButton(2, false);
|
||||||
|
|
||||||
|
CALL_AND_EXIT_ON_ERROR(httpd_resp_set_hdr, req, "Location", "/")
|
||||||
|
CALL_AND_EXIT(esphttpdutils::webserver_resp_send, req, esphttpdutils::ResponseStatus::TemporaryRedirect, "text/html", "Ok, continue at <a href=\"/\">/</a>")
|
||||||
|
}
|
||||||
|
else if (button == "profile3")
|
||||||
|
{
|
||||||
|
InputDispatcher::profileButton(3, true);
|
||||||
|
InputDispatcher::profileButton(3, false);
|
||||||
|
|
||||||
|
CALL_AND_EXIT_ON_ERROR(httpd_resp_set_hdr, req, "Location", "/")
|
||||||
|
CALL_AND_EXIT(esphttpdutils::webserver_resp_send, req, esphttpdutils::ResponseStatus::TemporaryRedirect, "text/html", "Ok, continue at <a href=\"/\">/</a>")
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
const auto msg = fmt::format("invalid {} {}", buttonParamName, button);
|
||||||
|
ESP_LOGW(TAG, "%.*s", msg.size(), msg.data());
|
||||||
|
CALL_AND_EXIT(esphttpdutils::webserver_resp_send, req, esphttpdutils::ResponseStatus::BadRequest, "text/plain", msg);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
esp_err_t webserver_triggerItem_handler(httpd_req_t *req)
|
||||||
|
{
|
||||||
|
std::string query;
|
||||||
|
if (auto result = esphttpdutils::webserver_get_query(req))
|
||||||
|
query = *result;
|
||||||
|
else
|
||||||
|
{
|
||||||
|
ESP_LOGE(TAG, "%.*s", result.error().size(), result.error().data());
|
||||||
|
CALL_AND_EXIT(esphttpdutils::webserver_resp_send, req, esphttpdutils::ResponseStatus::BadRequest, "text/plain", result.error());
|
||||||
|
}
|
||||||
|
|
||||||
|
std::size_t index;
|
||||||
|
|
||||||
|
constexpr const std::string_view indexParamName{"index"};
|
||||||
|
|
||||||
|
{
|
||||||
|
char valueBufEncoded[256];
|
||||||
|
if (const auto result = httpd_query_key_value(query.data(), indexParamName.data(), valueBufEncoded, 256); result != ESP_OK)
|
||||||
|
{
|
||||||
|
if (result == ESP_ERR_NOT_FOUND)
|
||||||
|
{
|
||||||
|
const auto msg = fmt::format("{} not set", indexParamName);
|
||||||
|
ESP_LOGW(TAG, "%.*s", msg.size(), msg.data());
|
||||||
|
CALL_AND_EXIT(esphttpdutils::webserver_resp_send, req, esphttpdutils::ResponseStatus::BadRequest, "text/plain", msg);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
const auto msg = fmt::format("httpd_query_key_value() {} failed with {}", indexParamName, esp_err_to_name(result));
|
||||||
|
ESP_LOGE(TAG, "%.*s", msg.size(), msg.data());
|
||||||
|
CALL_AND_EXIT(esphttpdutils::webserver_resp_send, req, esphttpdutils::ResponseStatus::BadRequest, "text/plain", msg);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
char valueBuf[257];
|
||||||
|
esphttpdutils::urldecode(valueBuf, valueBufEncoded);
|
||||||
|
|
||||||
|
std::string_view value{valueBuf};
|
||||||
|
|
||||||
|
if (auto parsed = cpputils::fromString<typeof(index)>(value))
|
||||||
|
{
|
||||||
|
index = *parsed;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
const auto msg = fmt::format("could not parse {} {}", indexParamName, value);
|
||||||
|
ESP_LOGW(TAG, "%.*s", msg.size(), msg.data());
|
||||||
|
CALL_AND_EXIT(esphttpdutils::webserver_resp_send, req, esphttpdutils::ResponseStatus::BadRequest, "text/plain", msg);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!currentDisplay)
|
||||||
|
{
|
||||||
|
constexpr const std::string_view msg = "currentDisplay is null";
|
||||||
|
ESP_LOGW(TAG, "%.*s", msg.size(), msg.data());
|
||||||
|
CALL_AND_EXIT(esphttpdutils::webserver_resp_send, req, esphttpdutils::ResponseStatus::BadRequest, "text/plain", msg);
|
||||||
|
}
|
||||||
|
|
||||||
|
auto *menuDisplay = currentDisplay->asMenuDisplay();
|
||||||
|
if (!menuDisplay)
|
||||||
|
{
|
||||||
|
constexpr const std::string_view msg = "currentDisplay is not a menu display";
|
||||||
|
ESP_LOGW(TAG, "%.*s", msg.size(), msg.data());
|
||||||
|
CALL_AND_EXIT(esphttpdutils::webserver_resp_send, req, esphttpdutils::ResponseStatus::BadRequest, "text/plain", msg);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (/*index < 0 ||*/ index >= menuDisplay->menuItemCount())
|
||||||
|
{
|
||||||
|
const auto msg = fmt::format("{} {} out of range (must be smaller than {})", indexParamName, index, menuDisplay->menuItemCount());
|
||||||
|
ESP_LOGW(TAG, "%.*s", msg.size(), msg.data());
|
||||||
|
CALL_AND_EXIT(esphttpdutils::webserver_resp_send, req, esphttpdutils::ResponseStatus::BadRequest, "text/plain", msg);
|
||||||
|
}
|
||||||
|
|
||||||
|
menuDisplay->getMenuItem(index).triggered();
|
||||||
|
|
||||||
|
CALL_AND_EXIT_ON_ERROR(httpd_resp_set_hdr, req, "Location", "/")
|
||||||
|
CALL_AND_EXIT(esphttpdutils::webserver_resp_send, req, esphttpdutils::ResponseStatus::TemporaryRedirect, "text/html", "Ok, continue at <a href=\"/\">/</a>")
|
||||||
|
}
|
||||||
|
|
||||||
|
esp_err_t webserver_setValue_handler(httpd_req_t *req)
|
||||||
|
{
|
||||||
|
std::string query;
|
||||||
|
if (auto result = esphttpdutils::webserver_get_query(req))
|
||||||
|
query = *result;
|
||||||
|
else
|
||||||
|
{
|
||||||
|
ESP_LOGE(TAG, "%.*s", result.error().size(), result.error().data());
|
||||||
|
CALL_AND_EXIT(esphttpdutils::webserver_resp_send, req, esphttpdutils::ResponseStatus::BadRequest, "text/plain", result.error());
|
||||||
|
}
|
||||||
|
|
||||||
|
int newValue;
|
||||||
|
|
||||||
|
constexpr const std::string_view valueParamName{"value"};
|
||||||
|
|
||||||
|
{
|
||||||
|
char valueBufEncoded[256];
|
||||||
|
if (const auto result = httpd_query_key_value(query.data(), valueParamName.data(), valueBufEncoded, 256); result != ESP_OK)
|
||||||
|
{
|
||||||
|
if (result == ESP_ERR_NOT_FOUND)
|
||||||
|
{
|
||||||
|
const auto msg = fmt::format("{} not set", valueParamName);
|
||||||
|
ESP_LOGW(TAG, "%.*s", msg.size(), msg.data());
|
||||||
|
CALL_AND_EXIT(esphttpdutils::webserver_resp_send, req, esphttpdutils::ResponseStatus::BadRequest, "text/plain", msg);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
const auto msg = fmt::format("httpd_query_key_value() {} failed with {}", valueParamName, esp_err_to_name(result));
|
||||||
|
ESP_LOGE(TAG, "%.*s", msg.size(), msg.data());
|
||||||
|
CALL_AND_EXIT(esphttpdutils::webserver_resp_send, req, esphttpdutils::ResponseStatus::BadRequest, "text/plain", msg);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
char valueBuf[257];
|
||||||
|
esphttpdutils::urldecode(valueBuf, valueBufEncoded);
|
||||||
|
|
||||||
|
std::string_view value{valueBuf};
|
||||||
|
|
||||||
|
if (auto parsed = cpputils::fromString<typeof(newValue)>(value))
|
||||||
|
{
|
||||||
|
newValue = *parsed;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
const auto msg = fmt::format("could not parse {} {}", valueParamName, value);
|
||||||
|
ESP_LOGW(TAG, "%.*s", msg.size(), msg.data());
|
||||||
|
CALL_AND_EXIT(esphttpdutils::webserver_resp_send, req, esphttpdutils::ResponseStatus::BadRequest, "text/plain", msg);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!currentDisplay)
|
||||||
|
{
|
||||||
|
constexpr const std::string_view msg = "currentDisplay is null";
|
||||||
|
ESP_LOGW(TAG, "%.*s", msg.size(), msg.data());
|
||||||
|
CALL_AND_EXIT(esphttpdutils::webserver_resp_send, req, esphttpdutils::ResponseStatus::BadRequest, "text/plain", msg);
|
||||||
|
}
|
||||||
|
|
||||||
|
auto *changeValueDisplay = currentDisplay->asChangeValueDisplayInterface();
|
||||||
|
if (!changeValueDisplay)
|
||||||
|
{
|
||||||
|
constexpr const std::string_view msg = "currentDisplay is not a change value display";
|
||||||
|
ESP_LOGW(TAG, "%.*s", msg.size(), msg.data());
|
||||||
|
CALL_AND_EXIT(esphttpdutils::webserver_resp_send, req, esphttpdutils::ResponseStatus::BadRequest, "text/plain", msg);
|
||||||
|
}
|
||||||
|
|
||||||
|
changeValueDisplay->setShownValue(newValue);
|
||||||
|
|
||||||
|
CALL_AND_EXIT_ON_ERROR(httpd_resp_set_hdr, req, "Location", "/")
|
||||||
|
CALL_AND_EXIT(esphttpdutils::webserver_resp_send, req, esphttpdutils::ResponseStatus::TemporaryRedirect, "text/html", "Ok, continue at <a href=\"/\">/</a>")
|
||||||
|
}
|
||||||
|
} // namespace
|
||||||
|
|
||||||
|
#endif
|
88
main/webserver_ota.h
Normal file
88
main/webserver_ota.h
Normal file
@ -0,0 +1,88 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
// esp-idf includes
|
||||||
|
#ifdef FEATURE_WEBSERVER
|
||||||
|
#include <esp_http_server.h>
|
||||||
|
#endif
|
||||||
|
#include <esp_log.h>
|
||||||
|
|
||||||
|
// 3rdparty lib includes
|
||||||
|
#include <htmlbuilder.h>
|
||||||
|
#include <fmt/core.h>
|
||||||
|
#include <espcppmacros.h>
|
||||||
|
#include <esphttpdutils.h>
|
||||||
|
|
||||||
|
// local includes
|
||||||
|
#ifdef FEATURE_OTA
|
||||||
|
#include "ota.h"
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#if defined(FEATURE_WEBSERVER) && defined(FEATURE_OTA)
|
||||||
|
namespace {
|
||||||
|
esp_err_t webserver_ota_handler(httpd_req_t *req);
|
||||||
|
esp_err_t webserver_trigger_ota_handler(httpd_req_t *req);
|
||||||
|
} // namespace
|
||||||
|
|
||||||
|
using esphttpdutils::HtmlTag;
|
||||||
|
|
||||||
|
namespace {
|
||||||
|
esp_err_t webserver_ota_handler(httpd_req_t *req)
|
||||||
|
{
|
||||||
|
std::string body;
|
||||||
|
|
||||||
|
{
|
||||||
|
HtmlTag htmlTag{"html", body};
|
||||||
|
|
||||||
|
{
|
||||||
|
HtmlTag headTag{"head", body};
|
||||||
|
|
||||||
|
{
|
||||||
|
HtmlTag titleTag{"title", body};
|
||||||
|
body += "Update";
|
||||||
|
}
|
||||||
|
|
||||||
|
body += "<meta name=\"viewport\" content=\"width=device-width, initial-scale=1, shrink-to-fit=no\" />";
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
HtmlTag bodyTag{"body", body};
|
||||||
|
|
||||||
|
{
|
||||||
|
HtmlTag h1Tag{"h1", body};
|
||||||
|
body += "Update";
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
HtmlTag pTag{"p", body};
|
||||||
|
body += "<a href=\"/\">Display control</a> "
|
||||||
|
"<b>Update</b> "
|
||||||
|
"<a href=\"/settings\">Settings</a>";
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
HtmlTag formTag{"form", "action=\"/triggerOta\" method=\"GET\"", body};
|
||||||
|
HtmlTag fieldsetTag{"fieldset", body};
|
||||||
|
{
|
||||||
|
HtmlTag legendTag{"legend", body};
|
||||||
|
body += "Trigger Update";
|
||||||
|
}
|
||||||
|
|
||||||
|
body += fmt::format("<input type=\"text\" name=\"url\" value=\"{}\" required />", esphttpdutils::htmlentities(stringSettings.otaUrl));
|
||||||
|
|
||||||
|
{
|
||||||
|
HtmlTag buttonTag{"button", "type=\"submit\"", body};
|
||||||
|
body += "Go";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
CALL_AND_EXIT(esphttpdutils::webserver_resp_send, req, esphttpdutils::ResponseStatus::Ok, "text/html", body)
|
||||||
|
}
|
||||||
|
|
||||||
|
esp_err_t webserver_trigger_ota_handler(httpd_req_t *req)
|
||||||
|
{
|
||||||
|
CALL_AND_EXIT(esphttpdutils::webserver_resp_send, req, esphttpdutils::ResponseStatus::Ok, "text/plain", "not yet implemented")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#endif
|
92
main/webserver_settings.h
Normal file
92
main/webserver_settings.h
Normal file
@ -0,0 +1,92 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
// esp-idf includes
|
||||||
|
#ifdef FEATURE_WEBSERVER
|
||||||
|
#include <esp_http_server.h>
|
||||||
|
#endif
|
||||||
|
#include <esp_log.h>
|
||||||
|
|
||||||
|
// 3rdparty lib includes
|
||||||
|
#include <htmlbuilder.h>
|
||||||
|
#include <fmt/core.h>
|
||||||
|
#include <espcppmacros.h>
|
||||||
|
#include <esphttpdutils.h>
|
||||||
|
|
||||||
|
// local includes
|
||||||
|
#include "globals.h"
|
||||||
|
|
||||||
|
#ifdef FEATURE_WEBSERVER
|
||||||
|
namespace {
|
||||||
|
esp_err_t webserver_settings_handler(httpd_req_t *req);
|
||||||
|
esp_err_t webserver_save_settings_handler(httpd_req_t *req);
|
||||||
|
} // namespace
|
||||||
|
|
||||||
|
using esphttpdutils::HtmlTag;
|
||||||
|
|
||||||
|
namespace {
|
||||||
|
esp_err_t webserver_settings_handler(httpd_req_t *req)
|
||||||
|
{
|
||||||
|
std::string body;
|
||||||
|
|
||||||
|
{
|
||||||
|
HtmlTag htmlTag{"html", body};
|
||||||
|
|
||||||
|
{
|
||||||
|
HtmlTag headTag{"head", body};
|
||||||
|
|
||||||
|
{
|
||||||
|
HtmlTag titleTag{"title", body};
|
||||||
|
body += "Settings";
|
||||||
|
}
|
||||||
|
|
||||||
|
body += "<meta name=\"viewport\" content=\"width=device-width, initial-scale=1, shrink-to-fit=no\" />";
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
HtmlTag bodyTag{"body", body};
|
||||||
|
|
||||||
|
{
|
||||||
|
HtmlTag h1Tag{"h1", body};
|
||||||
|
body += "Settings";
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
HtmlTag pTag{"p", body};
|
||||||
|
body += "<a href=\"/\">Display control</a> "
|
||||||
|
#ifdef FEATURE_OTA
|
||||||
|
"<a href=\"/ota\">Update</a> "
|
||||||
|
#endif
|
||||||
|
"<b>Settings</b>";
|
||||||
|
}
|
||||||
|
|
||||||
|
stringSettings.executeForEveryCommonSetting([&](const char *key, auto value){
|
||||||
|
HtmlTag formTag{"form", "action=\"/saveSettings\" method=\"GET\"", body};
|
||||||
|
HtmlTag fieldsetTag{"fieldset", body};
|
||||||
|
{
|
||||||
|
HtmlTag legendTag{"legend", body};
|
||||||
|
body += esphttpdutils::htmlentities(key);
|
||||||
|
}
|
||||||
|
|
||||||
|
body += fmt::format("<input type=\"text\" name=\"{}\" value=\"{}\" required />",
|
||||||
|
esphttpdutils::htmlentities(key),
|
||||||
|
esphttpdutils::htmlentities(value));
|
||||||
|
|
||||||
|
{
|
||||||
|
HtmlTag buttonTag{"button", "type=\"submit\"", body};
|
||||||
|
body += "Save";
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
CALL_AND_EXIT(esphttpdutils::webserver_resp_send, req, esphttpdutils::ResponseStatus::Ok, "text/html", body)
|
||||||
|
}
|
||||||
|
|
||||||
|
esp_err_t webserver_save_settings_handler(httpd_req_t *req)
|
||||||
|
{
|
||||||
|
CALL_AND_EXIT(esphttpdutils::webserver_resp_send, req, esphttpdutils::ResponseStatus::Ok, "text/plain", "not yet implemented")
|
||||||
|
}
|
||||||
|
} // namespace
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
Reference in New Issue
Block a user