#pragma once #include #include "driver/rmt_tx.h" #include "esp_log.h" #include "esp_system.h" #include "freertos/FreeRTOS.h" #include "freertos/task.h" #include "led_strip_encoder/led_strip_encoder.h" #include "nvs.h" #include "nvs_flash.h" #include "sdkconfig.h" static const char *LIGHT_TAG = "DisplayAmbientLight_Light"; #define RMT_TX_GPIO 1 #define STRIP_LED_NUMBER CONFIG_NUMBER_OF_LEDS #define EXAMPLE_CHASE_SPEED_MS (10) #define RMT_LED_STRIP_RESOLUTION_HZ 10000000 typedef enum light_mode_e { light_mode_init = 0, light_mode_connection_wifi = 1, light_mode_idle = 2, light_mode_mqtt_connected = 3, light_mode_desktop_online = 4, light_mode_desktop_sending_colors = 5, light_mode_off = 6, } light_mode_t; rmt_channel_handle_t led_chan = NULL; static uint8_t led_strip_pixels[STRIP_LED_NUMBER * 3]; rmt_encoder_handle_t led_encoder = NULL; rmt_transmit_config_t tx_config = { .loop_count = 0, // no transfer loop }; light_mode_t light_mode; float display_ambient_light_brightness = 1; uint8_t display_ambient_lighting_level = 255; float led_strip_red_calibration = 1.0; float led_strip_green_calibration = 1.0; float led_strip_blue_calibration = 1.0; void led_strip_fade_in_light_level(void *pvParameter) { float target = (float)display_ambient_lighting_level / 255.0; float step_length = (target - display_ambient_light_brightness) / 40.0; for (int t = 0; t < 40; t++) { display_ambient_light_brightness += step_length; vTaskDelay(pdMS_TO_TICKS(10)); } display_ambient_light_brightness = target; vTaskDelete(NULL); } void led_strip_set_brightness(uint8_t level) { if (display_ambient_lighting_level == level) { return; } display_ambient_lighting_level = level; xTaskCreate(led_strip_fade_in_light_level, "LED_STRIP_FADE_IN_LIGHT_LEVEL", 4096, NULL, 1, NULL); nvs_handle_t nvs_handle; esp_err_t err = nvs_open("storage", NVS_READWRITE, &nvs_handle); err = nvs_set_u8(nvs_handle, "brightness", level); if (err != ESP_OK) { ESP_LOGW(LIGHT_TAG, "Error (%s) saving light level!\n", esp_err_to_name(err)); nvs_close(nvs_handle); return; } err = nvs_commit(nvs_handle); if (err != ESP_OK) { ESP_LOGW(LIGHT_TAG, "Error (%s) saving light level!\n", esp_err_to_name(err)); } nvs_close(nvs_handle); } void led_strip_set_color_calibration(float red, float green, float blue) { led_strip_red_calibration = red; led_strip_green_calibration = green; led_strip_blue_calibration = blue; ESP_LOGI(LIGHT_TAG, "Calibration: %f %f %f", red, green, blue); nvs_handle_t local_nvs_handle; esp_err_t err = nvs_open("storage", NVS_READWRITE, &local_nvs_handle); if (err != ESP_OK) { ESP_LOGW(LIGHT_TAG, "Error (%s) opening NVS handle!\n", esp_err_to_name(err)); return; } err = nvs_set_u8(local_nvs_handle, "calibration_r", (uint32_t)(red * 255)); if (err != ESP_OK) { nvs_close(local_nvs_handle); ESP_LOGW(LIGHT_TAG, "Error (%s) write calibration_r failed!", esp_err_to_name(err)); return; } err = nvs_set_u8(local_nvs_handle, "calibration_g", (uint8_t)(green * 255)); if (err != ESP_OK) { nvs_close(local_nvs_handle); ESP_LOGW(LIGHT_TAG, "Error (%s) calibration_g failed!", esp_err_to_name(err)); return; } err = nvs_set_u8(local_nvs_handle, "calibration_b", (uint8_t)(blue * 255)); if (err != ESP_OK) { nvs_close(local_nvs_handle); ESP_LOGW(LIGHT_TAG, "Error (%s) calibration_b failed!", esp_err_to_name(err)); return; } err = nvs_commit(local_nvs_handle); if (err != ESP_OK) { ESP_LOGW(LIGHT_TAG, "Error (%s) save led_strip_red_calibration failed!", esp_err_to_name(err)); } nvs_close(local_nvs_handle); } void light_strip_transmit_task(void *pvParameter) { while (1) { ESP_ERROR_CHECK(rmt_transmit(led_chan, led_encoder, led_strip_pixels, sizeof(led_strip_pixels), &tx_config)); vTaskDelay(pdMS_TO_TICKS(10)); } } /** * @brief Simple helper function, converting HSV color space to RGB color * space * * Wiki: https://en.wikipedia.org/wiki/HSL_and_HSV * */ void led_strip_hsv2rgb(uint32_t h, uint32_t s, uint32_t v, uint32_t *r, uint32_t *g, uint32_t *b) { h %= 360; // h -> [0,360] uint32_t rgb_max = v * 2.55f; uint32_t rgb_min = rgb_max * (100 - s) / 100.0f; uint32_t i = h / 60; uint32_t diff = h % 60; // RGB adjustment amount by hue uint32_t rgb_adj = (rgb_max - rgb_min) * diff / 60; switch (i) { case 0: *r = rgb_max; *g = rgb_min + rgb_adj; *b = rgb_min; break; case 1: *r = rgb_max - rgb_adj; *g = rgb_max; *b = rgb_min; break; case 2: *r = rgb_min; *g = rgb_max; *b = rgb_min + rgb_adj; break; case 3: *r = rgb_min; *g = rgb_max - rgb_adj; *b = rgb_max; break; case 4: *r = rgb_min + rgb_adj; *g = rgb_min; *b = rgb_max; break; default: *r = rgb_max; *g = rgb_min; *b = rgb_max - rgb_adj; break; } } // void update_desktop_connection_state() { // static uint8_t tick = 0; // bool beat = tick / 10 % 2 ? 1 : 0; // switch (light_mode) { // case light_mode_desktop_online: // if (beat) { // led_strip_pixels[0] = 0; // led_strip_pixels[1] = 0; // led_strip_pixels[2] = 0; // } // led_strip_pixels[3] = 10 // break; // case light_mode_mqtt_connected: // if (beat) { // ESP_ERROR_CHECK( // light_led_strip->set_pixel(light_led_strip, 0, 10, 10, 10)); // ESP_ERROR_CHECK( // light_led_strip->set_pixel(light_led_strip, 1, 10, 10, 10)); // } // ESP_ERROR_CHECK( // light_led_strip->set_pixel(light_led_strip, 2, 22, 22, 22)); // break; // case light_mode_idle: // if (beat) { // ESP_ERROR_CHECK( // light_led_strip->set_pixel(light_led_strip, 0, 77, 77, 77)); // ESP_ERROR_CHECK( // light_led_strip->set_pixel(light_led_strip, 1, 77, 77, 77)); // ESP_ERROR_CHECK( // light_led_strip->set_pixel(light_led_strip, 2, 77, 77, 77)); // } // break; // default: // break; // } // tick++; // } void light_for_init() { ESP_LOGI(LIGHT_TAG, "light_for_init"); memset(led_strip_pixels, 0, sizeof(led_strip_pixels)); nvs_handle local_nvs_handle; esp_err_t err = nvs_open("storage", NVS_READWRITE, &local_nvs_handle); if (err != ESP_OK) { ESP_LOGW(LIGHT_TAG, "Error (%s) opening NVS handle!", esp_err_to_name(err)); } uint8_t r = 255, g = 255, b = 255; uint8_t brightness = 200; err = nvs_get_u8(local_nvs_handle, "calibration_r", &r); if (err != ESP_OK) { ESP_LOGW(LIGHT_TAG, "Error (%s) reading calibration_r!", esp_err_to_name(err)); } err = nvs_get_u8(local_nvs_handle, "calibration_g", &g); if (err != ESP_OK) { ESP_LOGW(LIGHT_TAG, "Error (%s) reading calibration_g!", esp_err_to_name(err)); } err = nvs_get_u8(local_nvs_handle, "calibration_b", &b); if (err != ESP_OK) { ESP_LOGW(LIGHT_TAG, "Error (%s) reading calibration_b!", esp_err_to_name(err)); } err = nvs_get_u8(local_nvs_handle, "brightness", &brightness); if (err != ESP_OK) { ESP_LOGW(LIGHT_TAG, "Error (%s) reading brightness!", esp_err_to_name(err)); } nvs_close(local_nvs_handle); // set brightness led_strip_set_brightness(brightness); // play init light float r_f = (float)r / 255.0; float g_f = (float)g / 255.0; float b_f = (float)b / 255.0; uint8_t init_r, init_g, init_b; do { for (uint8_t i = 0; i < 200; i++) { init_r = (uint8_t)(r_f * (float)i); init_g = (uint8_t)(g_f * (float)i); init_b = (uint8_t)(b_f * (float)i); for (int j = 0; j < STRIP_LED_NUMBER; j++) { led_strip_pixels[j * 3 + 0] = init_g; led_strip_pixels[j * 3 + 1] = init_r; led_strip_pixels[j * 3 + 2] = init_b; } vTaskDelay(pdMS_TO_TICKS(20)); } vTaskDelay(pdMS_TO_TICKS(100)); } while (light_mode == light_mode_init); led_strip_set_color_calibration((float)r / 255.0, (float)g / 255.0, (float)b / 255.0); } void light_for_connecting_wifi() { ESP_LOGI(LIGHT_TAG, "light_for_connecting_wifi"); int8_t tick_tock = 0; do { if (tick_tock) { led_strip_pixels[0] = 150; led_strip_pixels[1] = 150; led_strip_pixels[2] = 0; led_strip_pixels[3] = 200; led_strip_pixels[4] = 0; led_strip_pixels[5] = 0; } else { led_strip_pixels[0] = 200; led_strip_pixels[1] = 0; led_strip_pixels[2] = 0; led_strip_pixels[3] = 150; led_strip_pixels[4] = 150; led_strip_pixels[5] = 0; } tick_tock = !tick_tock; vTaskDelay(pdMS_TO_TICKS(200)); } while (light_mode == light_mode_connection_wifi); } void light_for_idle() { ESP_LOGI(LIGHT_TAG, "light_for_idle"); memset(led_strip_pixels, 0, sizeof(led_strip_pixels)); ESP_ERROR_CHECK(rmt_transmit(led_chan, led_encoder, led_strip_pixels, sizeof(led_strip_pixels), &tx_config)); uint32_t red = 0, green = 0, blue = 0; uint16_t step_length = 360 / STRIP_LED_NUMBER; for (uint16_t offset = 0; light_mode == light_mode_idle || light_mode == light_mode_mqtt_connected || light_mode == light_mode_desktop_online; offset = (offset + 1) % 360) { for (uint16_t j = 0, hue = offset; j < STRIP_LED_NUMBER; j++, hue += step_length) { // Build RGB values led_strip_hsv2rgb(hue, 50, 30, &red, &green, &blue); led_strip_pixels[j * 3 + 0] = green * display_ambient_light_brightness * led_strip_green_calibration; led_strip_pixels[j * 3 + 1] = red * display_ambient_light_brightness * led_strip_red_calibration; led_strip_pixels[j * 3 + 2] = blue * display_ambient_light_brightness * led_strip_blue_calibration; } // update_desktop_connection_state(); vTaskDelay(pdMS_TO_TICKS(10)); } } void light_strip_running_task(void *pv_parameters) { while (true) { if (!led_chan) { ESP_LOGE(LIGHT_TAG, "install WS2812 driver failed 2"); } switch (light_mode) { case light_mode_init: light_for_init(); break; case light_mode_connection_wifi: light_for_connecting_wifi(); break; case light_mode_idle: case light_mode_mqtt_connected: case light_mode_desktop_online: light_for_idle(); break; default: vTaskDelay(pdMS_TO_TICKS(100)); break; } } vTaskDelete(NULL); } void light_init_strip() { rmt_tx_channel_config_t tx_chan_config = { .clk_src = RMT_CLK_SRC_DEFAULT, // select source clock .gpio_num = RMT_TX_GPIO, .mem_block_symbols = 64, // increase the block size can make the LED less flickering .resolution_hz = RMT_LED_STRIP_RESOLUTION_HZ, .trans_queue_depth = 4, // set the number of transactions that can be // pending in the background }; ESP_ERROR_CHECK(rmt_new_tx_channel(&tx_chan_config, &led_chan)); ESP_LOGI(LIGHT_TAG, "Install led strip encoder"); led_strip_encoder_config_t encoder_config = { .resolution = RMT_LED_STRIP_RESOLUTION_HZ, }; ESP_ERROR_CHECK(rmt_new_led_strip_encoder(&encoder_config, &led_encoder)); ESP_LOGI(LIGHT_TAG, "Enable RMT TX channel"); ESP_ERROR_CHECK(rmt_enable(led_chan)); ESP_LOGI(LIGHT_TAG, "Start LED rainbow chase"); light_mode = light_mode_init; xTaskCreate(light_strip_running_task, "LIGHT_STRIP_RUNNING_TASK", 4096, NULL, 1, NULL); xTaskCreate(light_strip_transmit_task, "LIGHT_STRIP_TRANSMIT_TASK", 4096, NULL, 1, NULL); } void light_play_colors(uint16_t len, uint8_t *buffer) { // light_mode = light_mode_desktop_sending_colors; // uint16_t black_count = 0; // count of black pixels. r/g/b <= 10 // for (uint16_t led_index = 0, buffer_cursor = 0; // led_index < STRIP_LED_NUMBER && buffer_cursor < len; // led_index++, buffer_cursor += 3) { // uint8_t r = (uint8_t)((float)buffer[buffer_cursor] * // display_ambient_light_brightness * // led_strip_red_calibration), // g = (uint8_t)((float)buffer[buffer_cursor + 1] * // display_ambient_light_brightness * // led_strip_green_calibration), // b = (uint8_t)((float)buffer[buffer_cursor + 2] * // display_ambient_light_brightness * // led_strip_blue_calibration); // if (r <= 7 && g <= 7 && b <= 7) { // black_count++; // } // led_strip_pixels[led_index * 3 + 0] = g; // led_strip_pixels[led_index * 3 + 1] = r; // led_strip_pixels[led_index * 3 + 2] = b; // } // if (black_count > STRIP_LED_NUMBER / 5 * 4) { // uint8_t r = (uint8_t)((float)50 * display_ambient_light_brightness * // led_strip_red_calibration), // g = (uint8_t)((float)40 * display_ambient_light_brightness * // led_strip_green_calibration), // b = (uint8_t)((float)20 * display_ambient_light_brightness * // led_strip_blue_calibration); // for (uint16_t led_index = 0; led_index < STRIP_LED_NUMBER; led_index++) { // led_strip_pixels[led_index * 3 + 0] = g; // led_strip_pixels[led_index * 3 + 1] = r; // led_strip_pixels[led_index * 3 + 2] = b; // } // } vTaskDelay(pdMS_TO_TICKS(10)); } void light_play(light_mode_t mode) { ESP_LOGI(LIGHT_TAG, "light_play: %d", mode); light_mode = mode; } void set_display_ambient_light_colors(uint16_t offset, uint8_t *sub_pixels, uint16_t len) { // ESP_LOGI(LIGHT_TAG, "set_display_ambient_light_colors: offset: %d\t len: // %d", // offset, len); light_mode = light_mode_desktop_sending_colors; uint16_t black_count = 0; // count of black pixels. r/g/b <= 7 uint16_t global_end = len + offset * 3; if (global_end > STRIP_LED_NUMBER * 3) { global_end = STRIP_LED_NUMBER * 3; } for (uint16_t global_index = offset * 3, local_index = 0; global_index < global_end; global_index += 3, local_index += 3) { uint8_t r = (uint8_t)((float)sub_pixels[local_index + 0] * display_ambient_light_brightness * led_strip_red_calibration), g = (uint8_t)((float)sub_pixels[local_index + 1] * display_ambient_light_brightness * led_strip_green_calibration), b = (uint8_t)((float)sub_pixels[local_index + 2] * display_ambient_light_brightness * led_strip_blue_calibration); if (r <= 7 && g <= 7 && b <= 7) { black_count++; } led_strip_pixels[global_index + 0] = g; led_strip_pixels[global_index + 1] = r; led_strip_pixels[global_index + 2] = b; } if (black_count == len / 3) { uint8_t r = (uint8_t)((float)10 * display_ambient_light_brightness * led_strip_red_calibration), g = (uint8_t)((float)7 * display_ambient_light_brightness * led_strip_green_calibration), b = (uint8_t)((float)7 * display_ambient_light_brightness * led_strip_blue_calibration); for (uint16_t global_index = offset * 3; global_index < global_end; global_index += 3) { led_strip_pixels[global_index + 0] = g; led_strip_pixels[global_index + 1] = r; led_strip_pixels[global_index + 2] = b; } } }