Wednesday, May 27, 2020

Analog Clock using OLED and PIC Microcontroller

Hello Everyone, in this blog post I will show you guys how to make an analog clock on OLED SSD1306, this is a simple and cool project to display some basic animations on the OLED display.


We are going to build this project in the proteus simulation environment and use the PIC18F microcontroller. In terms of the software components, we need I2C Library, OLED Library, and DS1307 RTC Library.
If you are following my blogs you might be aware that these three libraries are already available and tested well, the following are the links to the previous posts, you can refer if you want.
OLED I2C Display Using Microchip PIC Microcontroller
Real Time Clock (DS1307) Interfacing with PIC Microcontroller
(My all new projects will be on GitHub and I am also trying to port all previous projects on GitHub so that these can be maintained well.)

The following are the steps to make an Analog Clock.
  • Make a small circle with a very small radius such as 1
  • Make a big circle with a big radius but make sure to not exceed the screen size.
  • Make Hour Ticks
    • We have 12 Hours and 360 degrees, which means 1 Hour equals 30 degrees.
    • We will use a for loop incremented by 30 degrees, to draw small lines representing ticks for each hour position. This is done by draw_clock_face function.
    • draw_clock_face function uses sine and cosine formulas to draw 12 marks of length 5 at each hour position.
  • Displaying Hour, Minute and Second Hands
    • Displaying timing using clock hands is very similar to displaying marks.
    • Hour Hand should be very small, while minute being medium and second hand should be longest of all.
  • Data from RTC is read and converted to decimal format from BCD format, and this is data is faced to the display_time function.
  • Executing this function per second will have an animation effect of moving the second hand of the clock.
  • In the project, we have called this per 500ms, the reason for this is the digital clock as we wanted to display the colon blinking, which is done half a second.
The following is the code and this can be downloaded or cloned from my GitHub page (click here).
#include "main.h"
#include "OLED.h"
#include "ds1307.h"
/* Project Related Macros */
#define CLOCK_CENTER_X ( 3u*OLED_Width()/4u )
#define CLOCK_CENTER_Y ( OLED_Height()/2u )
/* Local Variables */
static uint32_t millisecond = 0;
/* Local Function */
void timer0_init( void );
void system_init( void );
void display_clock_face( void );
void display_time( uint8_t, uint8_t, uint8_t);
/* Start From Here*/
void main(void)
{
uint32_t timestamp = 0;
char lcd_msg[16u] = {0};
system_init();
OLED_ClearDisplay();
OLED_Update();
while(1)
{
// Task-1 (500ms)
if( millis() - timestamp > 500ul )
{
uint8_t year, month, date, hour, minute, second;
static uint8_t sec_blink;
timestamp = millis();
OLED_ClearDisplay();
second = DS1307_Read (DS1307_SEC);
minute = DS1307_Read (DS1307_MIN);
hour = DS1307_Read (DS1307_HOUR);
date = DS1307_Read (DS1307_DATE);
month = DS1307_Read (DS1307_MONTH);
year = DS1307_Read (DS1307_YEAR);
// The value Read are in BCD Format, which needs to be converted into char
if( sec_blink )
{
sec_blink = FALSE;
sprintf (lcd_msg, "%c%c:%c%c:%c%c", \
BCD2UpperCh (hour), BCD2LowerCh (hour), \
BCD2UpperCh (minute), BCD2LowerCh (minute), \
BCD2UpperCh (second), BCD2LowerCh (second) );
}
else
{
sec_blink = TRUE;
sprintf (lcd_msg, "%c%c:%c%c %c%c", \
BCD2UpperCh (hour), BCD2LowerCh (hour), \
BCD2UpperCh (minute), BCD2LowerCh (minute), \
BCD2UpperCh (second), BCD2LowerCh (second) );
}
OLED_Write_Text (0, 0, " CLOCK");
OLED_Write_Text (4, 10, lcd_msg);
sprintf (lcd_msg, "%c%c/%c%c/%c%c", \
BCD2UpperCh (date), BCD2LowerCh (date), \
BCD2UpperCh (month), BCD2LowerCh (month), \
BCD2UpperCh (year), BCD2LowerCh (year) );
OLED_Write_Text (4, 18, lcd_msg);
// Analog Clock
display_clock_face();
display_time( BCD2Decimal(hour), \
BCD2Decimal(minute),\
BCD2Decimal(second));
OLED_Update ();
}
}
return;
}
/* Function Definitions */
/**
* @breif Initialize the system
*/
void system_init( void )
{
// Initialize 1ms Timer
timer0_init ();
// Initialize DS1307 and Start Time
DS1307_Init ();
// Initialize OLED
OLED_Init ();
Delay_ms(1000);
}
/**
* @brief Counts milliseconds till startup
* @return This function returns the number of milliseconds elapsed till the
* system is powered up.
*/
uint32_t millis( void )
{
uint32_t time = 0;
GIE = 0;
time = millisecond;
GIE = 1;
return time;
}
/**
* @brief Initialize Timer0 Module to generate 1ms interrupt
*/
void timer0_init( void )
{
// Code Generated Using mikroC Timer Calculator
T0CON = 0x88;
TMR0H = 0xD1;
TMR0L = 0x20;
ENABLE_INT();
TMR0IE = 1;
}
/**
* @brief PIC18F Interrupt Service Routine
*/
void __interrupt() ISR_ROUTINE( void )
{
if( TMR0IF )
{
TMR0IF = 0;
TMR0H = 0xD1;
TMR0L = 0x20;
millisecond++;
}
}
/**
* @brief Draw the Clock Face
*/
void display_clock_face( void )
{
uint16_t idx = 0;
uint16_t x1, x2, y1, y2;
// Draw Clock Face
// for( idx = 0; idx<2u; idx++ )
// {
// OLED_Circle( CLOCK_CENTER_X, CLOCK_CENTER_Y, 31-idx, 1); // Big Circle
// }
// for( idx=0; idx<3u; idx++ )
// {
// OLED_Circle( CLOCK_CENTER_X, CLOCK_CENTER_Y, idx, 1); // Small Circle
// }
OLED_Circle( CLOCK_CENTER_X, CLOCK_CENTER_Y, 31, 1); // Big Circle
OLED_Circle( CLOCK_CENTER_X, CLOCK_CENTER_Y, 2u, 1); // Small Circle
// Draw Small Mark for Every Hour
// hour ticks
for( idx=0; idx<360; idx=idx+30 )
{
// Begin at 0 degree and stop at 360 degree
// To convert degree into radians, we have to multiply degree by (pi/180)
// degree (in radians) = degree * pi / 180
// degree (in radians) = degree / (180/pi)
// pi = 57.29577951 ( Python math.pi value )
float angle = idx ;
angle = (angle/57.29577951) ; //Convert degrees to radians
x1 = (CLOCK_CENTER_X + (sin(angle)*31));
y1 = (CLOCK_CENTER_Y - (cos(angle)*31));
x2 = (CLOCK_CENTER_X + (sin(angle)*(31-5)));
y2 = (CLOCK_CENTER_Y - (cos(angle)*(31-5)));
OLED_Line(x1, y1, x2, y2, 1);
}
}
/**
* @brief Display the time
* @param hour hour values in decimal
* @param minute minute values in decimal
* @param second second values in decimal
*/
void display_time( uint8_t hour, uint8_t minute, uint8_t second)
{
float angle = 0.0;
uint16_t temp = 0;
uint16_t x, y;
// display second hand
temp = (uint16_t)second * (uint16_t)6u;
angle= ((float)temp/57.29577951) ; //Convert degrees to radians
x = ( CLOCK_CENTER_X + (sin(angle)*(29)) );
y = ( CLOCK_CENTER_Y - (cos(angle)*(29)) );
OLED_Line( CLOCK_CENTER_X, CLOCK_CENTER_Y, x, y, 1);
// display minute hand
temp = minute * 6u;
angle= ((float)temp/57.29577951) ; //Convert degrees to radians
x = ( CLOCK_CENTER_X + (sin(angle)*(22)) );
y = ( CLOCK_CENTER_Y - (cos(angle)*(22)) );
OLED_Line( CLOCK_CENTER_X, CLOCK_CENTER_Y, x, y, 1);
// display hour hand
angle = hour * 30 + ((minute*6)/12 ) ;
angle=(angle/57.29577951) ; //Convert degrees to radians
x = ( CLOCK_CENTER_X + (sin(angle)*(15)) );
y = ( CLOCK_CENTER_Y - (cos(angle)*(15)) );
OLED_Line( CLOCK_CENTER_X, CLOCK_CENTER_Y, x, y, 1);
}

The following is the simulation video link.

In case of any doubt, feel free to contact us.

1 comment: