hardware:clock
Различия
Показаны различия между двумя версиями страницы.
Предыдущая версия справа и слеваПредыдущая версияСледующая версия | Предыдущая версия | ||
hardware:clock [01.07.2018 08:59] – [Код] viacheslav | hardware:clock [30.07.2024 19:21] (текущий) – внешнее изменение 127.0.0.1 | ||
---|---|---|---|
Строка 1: | Строка 1: | ||
+ | ==== Железо ==== | ||
+ | [[https:// | ||
+ | [[https:// | ||
+ | С этим не работает, | ||
+ | ==== Подключение ==== | ||
+ | Схема выводов NodeMCU:\\ | ||
+ | {{: | ||
+ | |||
+ | Дисплей: | ||
+ | {{: | ||
+ | |||
+ | ==== Прошивка и ПО ==== | ||
+ | Драйвер - https:// | ||
+ | |||
+ | Прошиватор, | ||
+ | Запускать Win64\Release\ESP8266Flasher.exe.\\ | ||
+ | Arduino IDE: https:// | ||
+ | |||
+ | Arduino IDE -> Preferences -> Boards Manager URLs, добавить ссылку\\ | ||
+ | http:// | ||
+ | Далее Tools -> Boards -> Boards Manager\\ | ||
+ | В поле поиска вставьте esp8266, выберите ESP8266 Community и нажмите кнопку Установить | ||
+ | |||
+ | [[https:// | ||
+ | ==== Код ==== | ||
+ | Взят отсюда: | ||
+ | |||
+ | В Arduino IDE нужно скачать библиотеку поддержки дисплея Grove 4-digit display by Seeed Studio (поиск по 1637), библиотеку WiFiManager (её можно закомментировать в коде и не ставить), | ||
+ | |||
+ | Изменения: | ||
+ | * Часовой пояс изменён на московский (+3) | ||
+ | * NTP-сервер заменён на pool.ntp.org | ||
+ | * Ноги для дисплея заменены на CLK D6 (GPIO12), DIO D5 (GPIO14). | ||
+ | * Яркость дисплея выставлена на BRIGHT_TYPICAL (2 из диапазона 0-7) | ||
+ | * Синхронизация с сервером - раз в 129600 сек (36 часов), | ||
+ | |||
+ | <file lua ntp-clock.ino> | ||
+ | // | ||
+ | // | ||
+ | // This is a simple program which uses WiFi & an 4x7-segment display | ||
+ | // to show the current time, complete with blinking ":" | ||
+ | // | ||
+ | // Steve | ||
+ | // -- | ||
+ | // | ||
+ | |||
+ | |||
+ | // | ||
+ | // WiFi & over the air updates | ||
+ | // | ||
+ | #include < | ||
+ | #include < | ||
+ | |||
+ | // | ||
+ | // NTP uses UDP. | ||
+ | // | ||
+ | #include < | ||
+ | |||
+ | // | ||
+ | // The display-interface | ||
+ | // | ||
+ | #include " | ||
+ | |||
+ | // | ||
+ | // Time & NTP | ||
+ | // | ||
+ | #include " | ||
+ | |||
+ | |||
+ | // | ||
+ | // The name of this project. | ||
+ | // | ||
+ | // Used for: | ||
+ | // | ||
+ | // OTA name. | ||
+ | // | ||
+ | #define PROJECT_NAME " | ||
+ | |||
+ | |||
+ | // | ||
+ | // The timezone - comment out to stay at GMT. | ||
+ | // | ||
+ | #define TIME_ZONE (+3) | ||
+ | |||
+ | // | ||
+ | // Should we enable debugging (via serial-console output) ? | ||
+ | // | ||
+ | // Use either `#undef DEBUG`, or `#define DEBUG`. | ||
+ | // | ||
+ | #define DEBUG | ||
+ | |||
+ | |||
+ | // | ||
+ | // If we did then DEBUG_LOG will log a string, otherwise | ||
+ | // it will be ignored as a comment. | ||
+ | // | ||
+ | #ifdef DEBUG | ||
+ | # define DEBUG_LOG(x) Serial.print(x) | ||
+ | #else | ||
+ | # define DEBUG_LOG(x) | ||
+ | #endif | ||
+ | |||
+ | |||
+ | |||
+ | |||
+ | // | ||
+ | // Use the user-friendly WiFI library? | ||
+ | // | ||
+ | // If you do not want to use this then comment out the following line: | ||
+ | // | ||
+ | #define WIFI_MANAGER | ||
+ | |||
+ | // | ||
+ | // Otherwise define a SSID / Password | ||
+ | // | ||
+ | #ifdef WIFI_MANAGER | ||
+ | # include " | ||
+ | #else | ||
+ | # define WIFI_SSID " | ||
+ | # define WIFI_PASS " | ||
+ | #endif | ||
+ | |||
+ | |||
+ | // | ||
+ | // UDP-socket & local-port for replies. | ||
+ | // | ||
+ | WiFiUDP Udp; | ||
+ | unsigned int localPort = 2390; | ||
+ | |||
+ | |||
+ | // | ||
+ | // The NTP-server we use. | ||
+ | // | ||
+ | static const char ntpServerName[] = " | ||
+ | |||
+ | |||
+ | // | ||
+ | // Pin definitions for TM1637 and can be changed to other ports | ||
+ | // | ||
+ | #define CLK D6 | ||
+ | #define DIO D5 | ||
+ | TM1637 tm1637(CLK, DIO); | ||
+ | |||
+ | |||
+ | |||
+ | // | ||
+ | // This function is called when the device is powered-on. | ||
+ | // | ||
+ | void setup() | ||
+ | { | ||
+ | // Enable our serial port. | ||
+ | Serial.begin(115200); | ||
+ | |||
+ | // initialize the display | ||
+ | tm1637.init(); | ||
+ | |||
+ | // We want to see ":" | ||
+ | tm1637.point(true); | ||
+ | |||
+ | // | ||
+ | // Set the intensity - valid choices include: | ||
+ | // | ||
+ | // | ||
+ | // | ||
+ | // | ||
+ | // | ||
+ | tm1637.set(BRIGHT_TYPICAL); | ||
+ | |||
+ | // | ||
+ | // Handle WiFi setup | ||
+ | // | ||
+ | #ifdef WIFI_MANAGER | ||
+ | |||
+ | WiFiManager wifiManager; | ||
+ | wifiManager.autoConnect(PROJECT_NAME); | ||
+ | |||
+ | #else | ||
+ | // | ||
+ | // Connect to the WiFi network, and set a sane | ||
+ | // hostname so we can ping it by name. | ||
+ | // | ||
+ | WiFi.mode(WIFI_STA); | ||
+ | WiFi.hostname(PROJECT_NAME); | ||
+ | WiFi.begin(WIFI_SSID, | ||
+ | |||
+ | // | ||
+ | // Show that we are connecting to the WiFi. | ||
+ | // | ||
+ | DEBUG_LOG(" | ||
+ | |||
+ | // | ||
+ | // Try to connect to WiFi, constantly. | ||
+ | // | ||
+ | while (WiFi.status() != WL_CONNECTED) | ||
+ | { | ||
+ | DEBUG_LOG(" | ||
+ | delay(500); | ||
+ | } | ||
+ | |||
+ | // | ||
+ | // Now we are connected show the local IP address. | ||
+ | // | ||
+ | DEBUG_LOG(" | ||
+ | DEBUG_LOG(WiFi.localIP()); | ||
+ | DEBUG_LOG(" | ||
+ | |||
+ | #endif | ||
+ | |||
+ | // | ||
+ | // Ensure our UDP port is listening for receiving NTP-replies | ||
+ | // | ||
+ | Udp.begin(localPort); | ||
+ | |||
+ | |||
+ | // | ||
+ | // But now we are done, so we will setup the clock. | ||
+ | // | ||
+ | // We provide the time via NTP, and resync every 36 hours, | ||
+ | // as recommended at https:// | ||
+ | // This is more than enough in most cases. | ||
+ | // | ||
+ | setSyncProvider(getNtpTime); | ||
+ | setSyncInterval(129600); | ||
+ | |||
+ | // | ||
+ | // The final step is to allow over the air updates | ||
+ | // | ||
+ | // This is documented here: | ||
+ | // | ||
+ | // | ||
+ | // Hostname defaults to esp8266-[ChipID] | ||
+ | // | ||
+ | ArduinoOTA.setHostname(PROJECT_NAME); | ||
+ | |||
+ | ArduinoOTA.onStart([]() | ||
+ | { | ||
+ | DEBUG_LOG(" | ||
+ | }); | ||
+ | ArduinoOTA.onEnd([]() | ||
+ | { | ||
+ | DEBUG_LOG(" | ||
+ | }); | ||
+ | ArduinoOTA.onProgress([](unsigned int progress, unsigned int total) | ||
+ | { | ||
+ | char buf[16]; | ||
+ | memset(buf, ' | ||
+ | snprintf(buf, | ||
+ | DEBUG_LOG(buf); | ||
+ | DEBUG_LOG(" | ||
+ | }); | ||
+ | ArduinoOTA.onError([](ota_error_t error) | ||
+ | { | ||
+ | DEBUG_LOG(" | ||
+ | DEBUG_LOG(error); | ||
+ | DEBUG_LOG(" | ||
+ | |||
+ | if (error == OTA_AUTH_ERROR) | ||
+ | DEBUG_LOG(" | ||
+ | else if (error == OTA_BEGIN_ERROR) | ||
+ | DEBUG_LOG(" | ||
+ | else if (error == OTA_CONNECT_ERROR) | ||
+ | DEBUG_LOG(" | ||
+ | else if (error == OTA_RECEIVE_ERROR) | ||
+ | DEBUG_LOG(" | ||
+ | else if (error == OTA_END_ERROR) | ||
+ | DEBUG_LOG(" | ||
+ | }); | ||
+ | |||
+ | // | ||
+ | // Ensure the OTA process is running & listening. | ||
+ | // | ||
+ | ArduinoOTA.begin(); | ||
+ | |||
+ | |||
+ | } | ||
+ | |||
+ | |||
+ | |||
+ | // | ||
+ | // This function is called continously, | ||
+ | // for flashing the ":", | ||
+ | // | ||
+ | // We rely on the background NTP-updates to actually make sure | ||
+ | // that that works. | ||
+ | // | ||
+ | void loop() | ||
+ | { | ||
+ | static char buf[10] = { ' | ||
+ | static char prev[10] = { ' | ||
+ | static long last_read = 0; | ||
+ | static bool flash = true; | ||
+ | |||
+ | // | ||
+ | // Get the current hour/min | ||
+ | // | ||
+ | time_t nw = now(); | ||
+ | int cur_hour = hour(nw); | ||
+ | int cur_min | ||
+ | |||
+ | // | ||
+ | // Format them in a useful way. | ||
+ | // | ||
+ | sprintf(buf, | ||
+ | |||
+ | // | ||
+ | // If the current " | ||
+ | // that we displayed last loop .. | ||
+ | // | ||
+ | if (strcmp(buf, | ||
+ | { | ||
+ | // Update the display | ||
+ | tm1637.display(0, | ||
+ | tm1637.display(1, | ||
+ | tm1637.display(2, | ||
+ | tm1637.display(3, | ||
+ | |||
+ | // And cache it | ||
+ | strcpy(prev , buf); | ||
+ | |||
+ | } | ||
+ | |||
+ | |||
+ | // | ||
+ | // The preceeding piece of code would | ||
+ | // have ensured the display only updated | ||
+ | // when the hour/min changed. | ||
+ | // | ||
+ | // However note that we nuke the cached | ||
+ | // value every second - solely so we can | ||
+ | // blink the ":" | ||
+ | // | ||
+ | // Sigh | ||
+ | |||
+ | long now = millis(); | ||
+ | |||
+ | if ((last_read == 0) || | ||
+ | (abs(now - last_read) > 1000)) | ||
+ | { | ||
+ | // Invert the "show :" flag | ||
+ | flash = !flash; | ||
+ | |||
+ | // Apply it. | ||
+ | tm1637.point(flash); | ||
+ | |||
+ | // However note that the ":" | ||
+ | // unless/ | ||
+ | // force that to happen. | ||
+ | memset(prev, | ||
+ | last_read = now; | ||
+ | } | ||
+ | } | ||
+ | |||
+ | |||
+ | |||
+ | /*-------- NTP code ----------*/ | ||
+ | |||
+ | const int NTP_PACKET_SIZE = 48; // NTP time is in the first 48 bytes of message | ||
+ | byte packetBuffer[NTP_PACKET_SIZE]; | ||
+ | |||
+ | time_t getNtpTime() | ||
+ | { | ||
+ | IPAddress ntpServerIP; | ||
+ | |||
+ | // discard any previously received packets | ||
+ | while (Udp.parsePacket() > 0) ; | ||
+ | |||
+ | DEBUG_LOG(" | ||
+ | |||
+ | // get a random server from the pool | ||
+ | WiFi.hostByName(ntpServerName, | ||
+ | |||
+ | DEBUG_LOG(ntpServerName); | ||
+ | DEBUG_LOG(" | ||
+ | DEBUG_LOG(ntpServerIP); | ||
+ | DEBUG_LOG(" | ||
+ | |||
+ | sendNTPpacket(ntpServerIP); | ||
+ | |||
+ | delay(50); | ||
+ | uint32_t beginWait = millis(); | ||
+ | |||
+ | while ((millis() - beginWait) < 5000) | ||
+ | { | ||
+ | DEBUG_LOG("#" | ||
+ | int size = Udp.parsePacket(); | ||
+ | |||
+ | if (size >= NTP_PACKET_SIZE) | ||
+ | { | ||
+ | |||
+ | DEBUG_LOG(" | ||
+ | Udp.read(packetBuffer, | ||
+ | |||
+ | unsigned long secsSince1900; | ||
+ | |||
+ | // convert four bytes starting at location 40 to a long integer | ||
+ | secsSince1900 = (unsigned long)packetBuffer[40] << 24; | ||
+ | secsSince1900 |= (unsigned long)packetBuffer[41] << 16; | ||
+ | secsSince1900 |= (unsigned long)packetBuffer[42] << 8; | ||
+ | secsSince1900 |= (unsigned long)packetBuffer[43]; | ||
+ | |||
+ | // Now convert to the real time. | ||
+ | unsigned long now = secsSince1900 - 2208988800UL; | ||
+ | |||
+ | #ifdef TIME_ZONE | ||
+ | DEBUG_LOG(" | ||
+ | DEBUG_LOG(TIME_ZONE); | ||
+ | DEBUG_LOG(" | ||
+ | |||
+ | now += (TIME_ZONE * SECS_PER_HOUR); | ||
+ | #endif | ||
+ | |||
+ | return (now); | ||
+ | } | ||
+ | |||
+ | delay(50); | ||
+ | } | ||
+ | |||
+ | DEBUG_LOG(" | ||
+ | return 0; | ||
+ | } | ||
+ | |||
+ | // send an NTP request to the time server at the given address | ||
+ | void sendNTPpacket(IPAddress & | ||
+ | { | ||
+ | // set all bytes in the buffer to 0 | ||
+ | memset(packetBuffer, | ||
+ | |||
+ | // Initialize values needed to form NTP request | ||
+ | // (see URL above for details on the packets) | ||
+ | packetBuffer[0] = 0b11100011; | ||
+ | packetBuffer[1] = 0; // Stratum, or type of clock | ||
+ | packetBuffer[2] = 6; // Polling Interval | ||
+ | packetBuffer[3] = 0xEC; // Peer Clock Precision | ||
+ | |||
+ | // 8 bytes of zero for Root Delay & Root Dispersion | ||
+ | packetBuffer[12] = 49; | ||
+ | packetBuffer[13] = 0x4E; | ||
+ | packetBuffer[14] = 49; | ||
+ | packetBuffer[15] = 52; | ||
+ | |||
+ | // all NTP fields have been given values, now | ||
+ | // you can send a packet requesting a timestamp: | ||
+ | Udp.beginPacket(address, | ||
+ | Udp.write(packetBuffer, | ||
+ | Udp.endPacket(); | ||
+ | } | ||
+ | |||
+ | </ |