mirror of
https://github.com/k3ng/k3ng_rotator_controller.git
synced 2025-01-24 21:36:57 +00:00
340 lines
9.6 KiB
C++
340 lines
9.6 KiB
C++
|
// ---------------------------------------------------------------------------
|
|||
|
// Created by Francisco Malpartida on 20/08/11.
|
|||
|
// Copyright 2011 - Under creative commons license 3.0:
|
|||
|
// Attribution-ShareAlike CC BY-SA
|
|||
|
//
|
|||
|
// This software is furnished "as is", without technical support, and with no
|
|||
|
// warranty, express or implied, as to its usefulness for any purpose.
|
|||
|
//
|
|||
|
// Thread Safe: No
|
|||
|
// Extendable: No
|
|||
|
//
|
|||
|
// @file LCD_C0220BiZ.cpp
|
|||
|
// Display class implementation of the LCD API 1.0
|
|||
|
//
|
|||
|
// @brief Based on the LCD API 1.0 by dale@wentztech.com
|
|||
|
// This library implements the driver to any I2C display with the ST7036
|
|||
|
// LCD controller.
|
|||
|
// I2C displays based on the ST7032i should also be compatible.
|
|||
|
//
|
|||
|
// Other compatible displays:
|
|||
|
// - NHD‐C0220BiZ‐FSW‐FBW‐3V3M
|
|||
|
// - NHD-C0220BiZ-FS(RGB)-FBW-3VM
|
|||
|
// Non tested but should be compatible with no or little changes
|
|||
|
// - NHD-C0216CiZ-FSW-FBW-3V3
|
|||
|
// - NHD‐C0216CiZ‐FN‐FBW‐3V
|
|||
|
//
|
|||
|
//
|
|||
|
// @author F. Malpartida - fmalpartida@gmail.com
|
|||
|
// ---------------------------------------------------------------------------
|
|||
|
#include <Arduino.h> //all things wiring / arduino
|
|||
|
#include <Wire.h>
|
|||
|
#include <string.h> //needed for strlen()
|
|||
|
#include <inttypes.h>
|
|||
|
|
|||
|
#include "LCD.h"
|
|||
|
#include "ST7036.h"
|
|||
|
|
|||
|
// Class private constants and definition
|
|||
|
// ----------------------------------------------------------------------------
|
|||
|
const int CMD_DELAY = 1; // Command delay in miliseconds
|
|||
|
const int CHAR_DELAY = 0; // Delay between characters in miliseconds
|
|||
|
const int PIXEL_ROWS_PER_CHAR = 8; // Number of pixel rows in the LCD character
|
|||
|
const int MAX_USER_CHARS = 16; // Maximun number of user defined characters
|
|||
|
|
|||
|
// LCD Command set
|
|||
|
const uint8_t DISP_CMD = 0x0; // Command for the display
|
|||
|
const uint8_t RAM_WRITE_CMD = 0x40; // Write to display RAM
|
|||
|
const uint8_t CLEAR_DISP_CMD = 0x01; // Clear display command
|
|||
|
const uint8_t HOME_CMD = 0x02; // Set cursos at home (0,0)
|
|||
|
const uint8_t DISP_ON_CMD = 0x0C; // Display on command
|
|||
|
const uint8_t DISP_OFF_CMD = 0x08; // Display off Command
|
|||
|
const uint8_t SET_DDRAM_CMD = 0x80; // Set DDRAM address command
|
|||
|
const uint8_t CONTRAST_CMD = 0x70; // Set contrast LCD command
|
|||
|
const uint8_t FUNC_SET_TBL0 = 0x38; // Function set - 8 bit, 2 line display 5x8, inst table 0
|
|||
|
const uint8_t FUNC_SET_TBL1 = 0x39; // Function set - 8 bit, 2 line display 5x8, inst table 1
|
|||
|
|
|||
|
// LCD bitmap definition
|
|||
|
const uint8_t CURSOR_ON_BIT = ( 1 << 1 );// Cursor selection bit in Display on cmd.
|
|||
|
const uint8_t BLINK_ON_BIT = ( 1 << 0 );// Blink selection bit on Display on cmd.
|
|||
|
|
|||
|
// Driver DDRAM addressing
|
|||
|
const uint8_t dram_dispAddr [][3] =
|
|||
|
{
|
|||
|
{ 0x00, 0x00, 0x00 }, // One line display address
|
|||
|
{ 0x00, 0x40, 0x00 }, // Two line display address
|
|||
|
{ 0x00, 0x10, 0x20 } // Three line display address
|
|||
|
};
|
|||
|
|
|||
|
// Static member variable definitions
|
|||
|
// ----------------------------------------------------------------------------
|
|||
|
|
|||
|
// Static file scope variable definitions
|
|||
|
// ----------------------------------------------------------------------------
|
|||
|
|
|||
|
// Private support functions
|
|||
|
// ----------------------------------------------------------------------------
|
|||
|
|
|||
|
// CLASS METHODS
|
|||
|
// ----------------------------------------------------------------------------
|
|||
|
|
|||
|
// Constructors:
|
|||
|
// ---------------------------------------------------------------------------
|
|||
|
ST7036::ST7036(uint8_t num_lines, uint8_t num_col,
|
|||
|
uint8_t i2cAddr )
|
|||
|
{
|
|||
|
_num_lines = num_lines;
|
|||
|
_num_col = num_col;
|
|||
|
_i2cAddress = ( i2cAddr >> 1 );
|
|||
|
_cmdDelay = CMD_DELAY;
|
|||
|
_charDelay = CHAR_DELAY;
|
|||
|
_initialised = false;
|
|||
|
_backlightPin = -1;
|
|||
|
|
|||
|
}
|
|||
|
|
|||
|
ST7036::ST7036(uint8_t num_lines, uint8_t num_col,
|
|||
|
uint8_t i2cAddr, int8_t backlightPin )
|
|||
|
{
|
|||
|
_num_lines = num_lines;
|
|||
|
_num_col = num_col;
|
|||
|
_i2cAddress = ( i2cAddr >> 1 );
|
|||
|
_cmdDelay = CMD_DELAY;
|
|||
|
_charDelay = CHAR_DELAY;
|
|||
|
_initialised = false;
|
|||
|
_backlightPin = backlightPin;
|
|||
|
|
|||
|
// If there is a pin assigned to the BL, set it as an output
|
|||
|
// ---------------------------------------------------------
|
|||
|
if ( _backlightPin != 0 )
|
|||
|
{
|
|||
|
pinMode ( _backlightPin, OUTPUT );
|
|||
|
}
|
|||
|
|
|||
|
}
|
|||
|
|
|||
|
// Functions: modifiers (set), selectors (get) and class methods
|
|||
|
// ---------------------------------------------------------------------------
|
|||
|
void ST7036::init ()
|
|||
|
{
|
|||
|
size_t retVal;
|
|||
|
// Initialise the Wire library.
|
|||
|
Wire.begin();
|
|||
|
|
|||
|
Wire.beginTransmission ( _i2cAddress );
|
|||
|
Wire.write ( (byte)0x0 ); // Send command to the display
|
|||
|
Wire.write ( FUNC_SET_TBL0 );
|
|||
|
delay (10);
|
|||
|
Wire.write ( FUNC_SET_TBL1 );
|
|||
|
delay (10);
|
|||
|
Wire.write ( 0x14 ); // Set BIAS - 1/5
|
|||
|
Wire.write ( 0x73 ); // Set contrast low byte
|
|||
|
Wire.write ( 0x5E ); // ICON disp on, Booster on, Contrast high byte
|
|||
|
Wire.write ( 0x6D ); // Follower circuit (internal), amp ratio (6)
|
|||
|
Wire.write ( 0x0C ); // Display on
|
|||
|
Wire.write ( 0x01 ); // Clear display
|
|||
|
Wire.write ( 0x06 ); // Entry mode set - increment
|
|||
|
_status = Wire.endTransmission ();
|
|||
|
|
|||
|
if ( _status == 0 )
|
|||
|
{
|
|||
|
_initialised = true;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
void ST7036::setDelay (int cmdDelay,int charDelay)
|
|||
|
{
|
|||
|
_cmdDelay = cmdDelay;
|
|||
|
_charDelay = charDelay;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
void ST7036::command(uint8_t value)
|
|||
|
{
|
|||
|
// If the LCD has been initialised correctly, write to it
|
|||
|
if ( _initialised )
|
|||
|
{
|
|||
|
Wire.beginTransmission ( _i2cAddress );
|
|||
|
Wire.write ( DISP_CMD );
|
|||
|
Wire.write ( value );
|
|||
|
_status = Wire.endTransmission ();
|
|||
|
delay(_cmdDelay);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
size_t ST7036::write(uint8_t value)
|
|||
|
{
|
|||
|
// If the LCD has been initialised correctly write to it
|
|||
|
// -----------------------------------------------------
|
|||
|
if ( _initialised )
|
|||
|
{
|
|||
|
|
|||
|
// If it is a new line, set the cursor to the next line (1,0)
|
|||
|
// ----------------------------------------------------------
|
|||
|
if ( value == '\n' )
|
|||
|
{
|
|||
|
setCursor (1,0);
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
Wire.beginTransmission ( _i2cAddress );
|
|||
|
Wire.write ( RAM_WRITE_CMD );
|
|||
|
Wire.write ( value );
|
|||
|
_status = Wire.endTransmission ();
|
|||
|
delay(_charDelay);
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
size_t ST7036::write(const uint8_t *buffer, size_t size)
|
|||
|
{
|
|||
|
// If the LCD has been initialised correctly, write to it
|
|||
|
// ------------------------------------------------------
|
|||
|
if ( _initialised )
|
|||
|
{
|
|||
|
Wire.beginTransmission ( _i2cAddress );
|
|||
|
Wire.write ( RAM_WRITE_CMD );
|
|||
|
Wire.write ( (uint8_t *)buffer, size );
|
|||
|
_status = Wire.endTransmission ();
|
|||
|
delay(_charDelay);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
void ST7036::clear()
|
|||
|
{
|
|||
|
command (CLEAR_DISP_CMD);
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
void ST7036::home()
|
|||
|
{
|
|||
|
command ( HOME_CMD );
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
void ST7036::on()
|
|||
|
{
|
|||
|
command ( DISP_ON_CMD );
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
void ST7036::off()
|
|||
|
{
|
|||
|
command ( DISP_OFF_CMD );
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
void ST7036::cursor_on()
|
|||
|
{
|
|||
|
command ( DISP_ON_CMD | CURSOR_ON_BIT );
|
|||
|
}
|
|||
|
|
|||
|
void ST7036::cursor_off()
|
|||
|
{
|
|||
|
command ( DISP_ON_CMD & ~(CURSOR_ON_BIT) );
|
|||
|
}
|
|||
|
|
|||
|
void ST7036::blink_on()
|
|||
|
{
|
|||
|
command ( DISP_ON_CMD | BLINK_ON_BIT );
|
|||
|
}
|
|||
|
|
|||
|
void ST7036::blink_off()
|
|||
|
{
|
|||
|
command ( DISP_ON_CMD & ~(BLINK_ON_BIT) );
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
void ST7036::setCursor(uint8_t line_num, uint8_t x)
|
|||
|
{
|
|||
|
uint8_t base = 0x00;
|
|||
|
|
|||
|
// If the LCD has been initialised correctly, write to it
|
|||
|
// ------------------------------------------------------
|
|||
|
if ( _initialised )
|
|||
|
{
|
|||
|
// set the baseline address with respect to the number of lines of
|
|||
|
// the display
|
|||
|
base = dram_dispAddr[_num_lines-1][line_num];
|
|||
|
base = SET_DDRAM_CMD + base + x;
|
|||
|
command ( base );
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
#ifdef _LCDEXPANDED
|
|||
|
uint8_t ST7036::status(){
|
|||
|
|
|||
|
return _status;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
uint8_t ST7036::keypad ()
|
|||
|
{
|
|||
|
// NOT SUPPORTED
|
|||
|
return 0;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
void ST7036::load_custom_character (uint8_t char_num, uint8_t *rows)
|
|||
|
{
|
|||
|
// If the LCD has been initialised correctly start writing to it
|
|||
|
// --------------------------------------------------------------------------
|
|||
|
if ( _initialised )
|
|||
|
{
|
|||
|
// If it is a valid place holder for the character, write it into the
|
|||
|
// display's CGRAM
|
|||
|
// --------------------------------------------------------------------------
|
|||
|
if ( char_num < MAX_USER_CHARS )
|
|||
|
{
|
|||
|
// Set up the display to write into CGRAM - configure LCD to use func table 0
|
|||
|
Wire.beginTransmission ( _i2cAddress );
|
|||
|
Wire.write ( DISP_CMD );
|
|||
|
Wire.write ( FUNC_SET_TBL0 ); // Function set: 8 bit, 2 line display 5x8, funct tab 0
|
|||
|
delay ( _cmdDelay );
|
|||
|
|
|||
|
// Set CGRAM position to write
|
|||
|
Wire.write ( RAM_WRITE_CMD + (PIXEL_ROWS_PER_CHAR * char_num) );
|
|||
|
_status = Wire.endTransmission ();
|
|||
|
|
|||
|
// If we have changed the function table and configured the CGRAM position
|
|||
|
// write the new character to the LCD's CGRAM
|
|||
|
// -----------------------------------------------------------------------
|
|||
|
if ( _status == 0 )
|
|||
|
{
|
|||
|
write ( rows, PIXEL_ROWS_PER_CHAR ); // write the character to CGRAM
|
|||
|
|
|||
|
// Leave the LCD as it was - function table 1 DDRAM and set the cursor
|
|||
|
// position to (0, 0) to start writing.
|
|||
|
command ( FUNC_SET_TBL1 );
|
|||
|
setCursor ( 0,0 );
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
void ST7036::setBacklight(uint8_t new_val)
|
|||
|
{
|
|||
|
// Set analog write to the pin, the routine already checks if it can
|
|||
|
// set a PWM or not.
|
|||
|
// -----------------------------------------------------------------
|
|||
|
if ( _backlightPin != -1 )
|
|||
|
{
|
|||
|
analogWrite ( _backlightPin, new_val );
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
void ST7036::setContrast(uint8_t new_val)
|
|||
|
{
|
|||
|
// Only allow 15 levels of contrast
|
|||
|
new_val = map ( new_val, 0, 255, 0, 15 );
|
|||
|
|
|||
|
command(CONTRAST_CMD + new_val);
|
|||
|
}
|
|||
|
#endif // _LCDEXPANDED
|
|||
|
|