1
0
mirror of https://github.com/esphome/esphome.git synced 2025-10-26 20:53:50 +00:00
Files
esphome/esphome/components/tm1651/tm1651.cpp
mrtoy-me 88d8cfe6a2 [tm1651] Remove dependency on Arduino Library (#9645)
Co-authored-by: pre-commit-ci-lite[bot] <117423508+pre-commit-ci-lite[bot]@users.noreply.github.com>
Co-authored-by: Keith Burzinski <kbx81x@gmail.com>
2025-07-30 20:20:55 -05:00

261 lines
8.3 KiB
C++

// This Esphome TM1651 component for use with Mini Battery Displays (7 LED levels)
// and removes the Esphome dependency on the TM1651 Arduino library.
// It was largely based on the work of others as set out below.
// @mrtoy-me July 2025
// ==============================================================================================
// Original Arduino TM1651 library:
// Author:Fred.Chu
// Date:14 August, 2014
// Applicable Module: Battery Display v1.0
// This library is free software; you can redistribute it and/or
// modify it under the terms of the GNU Lesser General Public
// License as published by the Free Software Foundation; either
// version 2.1 of the License, or (at your option) any later version.
// This library is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.See the GNU
// Lesser General Public License for more details.
// Modified record:
// Author: Detlef Giessmann Germany
// Mail: mydiyp@web.de
// Demo for the new 7 LED Battery-Display 2017
// IDE: Arduino-1.6.5
// Type: OPEN-SMART CX10*4RY68 4Color
// Date: 01.05.2017
// ==============================================================================================
// Esphome component using arduino TM1651 library:
// MIT License
// Copyright (c) 2019 freekode
// ==============================================================================================
// Library and command-line (python) program to control mini battery displays on Raspberry Pi:
// MIT License
// Copyright (c) 2020 Koen Vervloese
// ==============================================================================================
// MIT License
// Permission is hereby granted, free of charge, to any person obtaining a copy
// of this software and associated documentation files (the "Software"), to deal
// in the Software without restriction, including without limitation the rights
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
// copies of the Software, and to permit persons to whom the Software is
// furnished to do so, subject to the following conditions:
// The above copyright notice and this permission notice shall be included in all
// copies or substantial portions of the Software.
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
// SOFTWARE.
#include "tm1651.h"
#include "esphome/core/log.h"
namespace esphome {
namespace tm1651 {
static const char *const TAG = "tm1651.display";
static const bool LINE_HIGH = true;
static const bool LINE_LOW = false;
// TM1651 maximum frequency is 500 kHz (duty ratio 50%) = 2 microseconds / cycle
static const uint8_t CLOCK_CYCLE = 8;
static const uint8_t HALF_CLOCK_CYCLE = CLOCK_CYCLE / 2;
static const uint8_t QUARTER_CLOCK_CYCLE = CLOCK_CYCLE / 4;
static const uint8_t ADDR_FIXED = 0x44; // fixed address mode
static const uint8_t ADDR_START = 0xC0; // address of the display register
static const uint8_t DISPLAY_OFF = 0x80;
static const uint8_t DISPLAY_ON = 0x88;
static const uint8_t MAX_DISPLAY_LEVELS = 7;
static const uint8_t PERCENT100 = 100;
static const uint8_t PERCENT50 = 50;
static const uint8_t TM1651_BRIGHTNESS_DARKEST = 0;
static const uint8_t TM1651_BRIGHTNESS_TYPICAL = 2;
static const uint8_t TM1651_BRIGHTNESS_BRIGHTEST = 7;
static const uint8_t TM1651_LEVEL_TAB[] = {0b00000000, 0b00000001, 0b00000011, 0b00000111,
0b00001111, 0b00011111, 0b00111111, 0b01111111};
// public
void TM1651Display::setup() {
this->clk_pin_->setup();
this->clk_pin_->pin_mode(gpio::FLAG_OUTPUT);
this->dio_pin_->setup();
this->dio_pin_->pin_mode(gpio::FLAG_OUTPUT);
this->brightness_ = TM1651_BRIGHTNESS_TYPICAL;
// clear display
this->display_level_();
this->update_brightness_(DISPLAY_ON);
}
void TM1651Display::dump_config() {
ESP_LOGCONFIG(TAG, "Battery Display");
LOG_PIN(" CLK: ", clk_pin_);
LOG_PIN(" DIO: ", dio_pin_);
}
void TM1651Display::set_brightness(uint8_t new_brightness) {
this->brightness_ = this->remap_brightness_(new_brightness);
if (this->display_on_) {
this->update_brightness_(DISPLAY_ON);
}
}
void TM1651Display::set_level(uint8_t new_level) {
if (new_level > MAX_DISPLAY_LEVELS)
new_level = MAX_DISPLAY_LEVELS;
this->level_ = new_level;
if (this->display_on_) {
this->display_level_();
}
}
void TM1651Display::set_level_percent(uint8_t percentage) {
this->level_ = this->calculate_level_(percentage);
if (this->display_on_) {
this->display_level_();
}
}
void TM1651Display::turn_off() {
this->display_on_ = false;
this->update_brightness_(DISPLAY_OFF);
}
void TM1651Display::turn_on() {
this->display_on_ = true;
// display level as it could have been changed when display turned off
this->display_level_();
this->update_brightness_(DISPLAY_ON);
}
// protected
uint8_t TM1651Display::calculate_level_(uint8_t percentage) {
if (percentage > PERCENT100)
percentage = PERCENT100;
// scale 0-100% to 0-7 display levels
// use integer arithmetic with rounding
uint16_t initial_scaling = (percentage * MAX_DISPLAY_LEVELS) + PERCENT50;
return (uint8_t) (initial_scaling / PERCENT100);
}
void TM1651Display::display_level_() {
this->start_();
this->write_byte_(ADDR_FIXED);
this->stop_();
this->start_();
this->write_byte_(ADDR_START);
this->write_byte_(TM1651_LEVEL_TAB[this->level_]);
this->stop_();
}
uint8_t TM1651Display::remap_brightness_(uint8_t new_brightness) {
if (new_brightness <= 1)
return TM1651_BRIGHTNESS_DARKEST;
if (new_brightness == 2)
return TM1651_BRIGHTNESS_TYPICAL;
// new_brightness >= 3
return TM1651_BRIGHTNESS_BRIGHTEST;
}
void TM1651Display::update_brightness_(uint8_t on_off_control) {
this->start_();
this->write_byte_(on_off_control | this->brightness_);
this->stop_();
}
// low level functions
bool TM1651Display::write_byte_(uint8_t data) {
// data bit written to DIO when CLK is low
for (uint8_t i = 0; i < 8; i++) {
this->half_cycle_clock_low_((bool) (data & 0x01));
this->half_cycle_clock_high_();
data >>= 1;
}
// start 9th cycle, setting DIO high and look for ack
this->half_cycle_clock_low_(LINE_HIGH);
return this->half_cycle_clock_high_ack_();
}
void TM1651Display::half_cycle_clock_low_(bool data_bit) {
// first half cycle, clock low and write data bit
this->clk_pin_->digital_write(LINE_LOW);
delayMicroseconds(QUARTER_CLOCK_CYCLE);
this->dio_pin_->digital_write(data_bit);
delayMicroseconds(QUARTER_CLOCK_CYCLE);
}
void TM1651Display::half_cycle_clock_high_() {
// second half cycle, clock high
this->clk_pin_->digital_write(LINE_HIGH);
delayMicroseconds(HALF_CLOCK_CYCLE);
}
bool TM1651Display::half_cycle_clock_high_ack_() {
// second half cycle, clock high and check for ack
this->clk_pin_->digital_write(LINE_HIGH);
delayMicroseconds(QUARTER_CLOCK_CYCLE);
this->dio_pin_->pin_mode(gpio::FLAG_INPUT);
// valid ack on DIO is low
bool ack = (!this->dio_pin_->digital_read());
this->dio_pin_->pin_mode(gpio::FLAG_OUTPUT);
// ack should be set DIO low by now
// if its not, set DIO low before the next cycle
if (!ack) {
this->dio_pin_->digital_write(LINE_LOW);
}
delayMicroseconds(QUARTER_CLOCK_CYCLE);
// begin next cycle
this->clk_pin_->digital_write(LINE_LOW);
return ack;
}
void TM1651Display::start_() {
// start data transmission
this->delineate_transmission_(LINE_HIGH);
}
void TM1651Display::stop_() {
// stop data transmission
this->delineate_transmission_(LINE_LOW);
}
void TM1651Display::delineate_transmission_(bool dio_state) {
// delineate data transmission
// DIO changes its value while CLK is high
this->dio_pin_->digital_write(dio_state);
delayMicroseconds(HALF_CLOCK_CYCLE);
this->clk_pin_->digital_write(LINE_HIGH);
delayMicroseconds(QUARTER_CLOCK_CYCLE);
this->dio_pin_->digital_write(!dio_state);
delayMicroseconds(QUARTER_CLOCK_CYCLE);
}
} // namespace tm1651
} // namespace esphome