// https://www.hackster.io/OriginalCaveman/max7219-conquers-old-school-16-segment-displays-61ba8a
#include "pgmspace.h"
#include "stdlib.h"
const int NUMBER_OF_ELEMENTS = 96;
const int MAX_SIZE = 10;
//Outer Segment Codes
const char OSeg [NUMBER_OF_ELEMENTS] [MAX_SIZE] PROGMEM = {
{"00000000"}, //
{"00000000"}, //
{"00000000"}, //
{"00000000"}, //
{"00000000"}, //
{"00000000"}, //
{"00000000"}, //
{"00000000"}, //
{"00000000"}, //
{"00000000"}, //
{"00000000"}, //*
{"00000000"}, //+
{"00000000"}, //
{"00000000"}, //-
{"00000000"}, //
{"00000000"}, ///
{"00111100"}, //0
{"00011000"}, //1
{"00110100"}, //2
{"00111100"}, //3
{"00011000"}, //4
{"00101100"}, //5
{"00101100"}, //6
{"00111000"}, //7
{"00111100"}, //8
{"00111100"}, //9
{"00000000"}, //
{"00000000"}, //
{"00000000"}, //
{"00000000"}, //
{"00000000"}, //
{"00000000"}, //
{"11110111"}, //@
{"11111001"}, //A
{"01111110"}, //B
{"11100111"}, //C
{"01111110"}, //D
{"11100111"}, //E
{"11100001"}, //F
{"11101111"}, //G
{"10011001"}, //H
{"01100110"}, //I
{"00011111"}, //J
{"10000001"}, //K
{"10000111"}, //L
{"10011001"}, //M
{"10011001"}, //N
{"11111111"}, //O
{"11110001"}, //P
{"11111111"}, //Q
{"11110001"}, //R
{"11101110"}, //S
{"01100000"}, //T
{"10011111"}, //U
{"10000001"}, //V
{"10011001"}, //W
{"00000000"}, //X
{"10011110"}, //Y
{"01100110"}, //Z
{"00100100"}, //[
{"00000000"}, //\
{"01000010"}, //]
{"00000000"}, //^
{"00000110"}, //_
{"00000000"}, //`
{"00000111"}, //a
{"10000011"}, //b
{"00000011"}, //c
{"00000011"}, //d
{"00000011"}, //e
{"00100000"}, //f
{"11000010"}, //g
{"10000001"}, //h
{"00000000"}, //i
{"00000011"}, //j
{"00000000"}, //k
{"00000000"}, //l
{"00001001"}, //m
{"00000001"}, //n
{"00000011"}, //o
{"11000001"}, //p
{"11000000"}, //q
{"00000001"}, //r
{"11000010"}, //s
{"10000011"}, //t
{"00000011"}, //u
{"00000001"}, //v
{"00001001"}, //w
{"00000000"}, //x
{"10000010"}, //y
{"00000010"}, //z
{"00100100"}, //{
{"00000000"}, //|
{"01000010"}, //}
{"00000000"}, //~
{"00000000"}, //
};
// Inner Segment Codes
const char ISeg [NUMBER_OF_ELEMENTS] [MAX_SIZE] PROGMEM = {
{"00000000"}, //
{"00000000"}, //
{"00000000"}, //
{"00000000"}, //
{"00000000"}, //
{"00000000"}, //
{"00000000"}, //
{"00000000"}, //
{"00000000"}, //
{"00000000"}, //
{"11111111"}, //*
{"10100101"}, //+
{"00000000"}, //
{"10000001"}, //-
{"00000000"}, //
{"00010010"}, ///
{"00100100"}, //0
{"00010000"}, //1
{"10000100"}, //2
{"10000000"}, //3
{"10100000"}, //4
{"10100000"}, //5
{"10100100"}, //6
{"00000000"}, //7
{"10100100"}, //8
{"10100000"}, //9
{"00000000"}, //
{"00000000"}, //
{"00000000"}, //
{"00000000"}, //
{"00000000"}, //
{"00000000"}, //
{"10100000"}, //@
{"10000001"}, //A
{"10100100"}, //B
{"00000000"}, //C
{"00100100"}, //D
{"00000001"}, //E
{"00000001"}, //F
{"10000000"}, //G
{"10000001"}, //H
{"00100100"}, //I
{"00000000"}, //J
{"00011001"}, //K
{"00000000"}, //L
{"01010000"}, //M
{"01001000"}, //N
{"00000000"}, //O
{"10000001"}, //P
{"00001000"}, //Q
{"10001001"}, //R
{"10000001"}, //S
{"00100100"}, //T
{"00000000"}, //U
{"00010010"}, //V
{"00001010"}, //W
{"01011010"}, //X
{"10000001"}, //Y
{"00010010"}, //Z
{"00100100"}, //[
{"01001000"}, //\
{"00100100"}, //]
{"00001010"}, //^
{"00000000"}, //_
{"01000000"}, //`
{"00000101"}, //a
{"00000101"}, //b
{"00000001"}, //c
{"00100101"}, //d
{"00000011"}, //e
{"10100101"}, //f
{"00100101"}, //g
{"00000101"}, //h
{"00000100"}, //i
{"00100100"}, //j
{"00111100"}, //k
{"00100100"}, //l
{"10000101"}, //m
{"00000101"}, //n
{"00000101"}, //o
{"00100001"}, //p
{"00100101"}, //q
{"00000001"}, //r
{"00000101"}, //s
{"00000001"}, //t
{"00000100"}, //u
{"00000010"}, //v
{"00001010"}, //w
{"01011010"}, //x
{"00100101"}, //y
{"00000011"}, //z
{"00100101"}, //{
{"00100100"}, //|
{"10100100"}, //}
{"10010011"}, //~
{"00000000"}, //
};
#define Interval_Message 1200
#define MAX7219_DATA 2
#define MAX7219_CLOCK 4
#define MAX7219_LOAD 3
#define LdInner 9
#define LdOuter 8
unsigned long time1 = 0; // timer function
int x = 0; // message counter
// Various 16 character strings to display
const char string_0[] PROGMEM = " PERSISTANCE OF ";
const char string_1[] PROGMEM = " VISION DEMO ";
const char string_2[] PROGMEM = "USES Arduino Uno";
const char string_3[] PROGMEM = " Some AND gates ";
const char string_4[] PROGMEM = "MAX7219 and 16- ";
const char string_5[] PROGMEM = "Segment displays";
const char string_6[] PROGMEM = " ";
// Load the above strings into progmem
const char *const string_table[] PROGMEM = {string_0, string_1, string_2, string_3, string_4, string_5, string_6};
// 16 characters + 1 null
char buffer[17];
void initialize() {
pinMode(MAX7219_LOAD, OUTPUT);
pinMode(MAX7219_CLOCK, OUTPUT);
pinMode(MAX7219_DATA, OUTPUT);
pinMode(LdInner, OUTPUT);
pinMode(LdOuter, OUTPUT);
digitalWrite(MAX7219_LOAD, LOW);
digitalWrite(LdInner, LOW);
digitalWrite(LdOuter, LOW);
}
// -------------------------------------------------------------------------------------------------------------------------------------------
// Note: the 'data' is 16 bits which are split into highByte (for U6) and lowByte (for U2)
void OuterSegments(byte address, word data) {
// AND Gate U1 sections A (clock signal) and D (load signal)
digitalWrite(LdOuter, HIGH);
// U2 first gets the high byte (Upper 8 Displays Outer Segments) from the UNO
shiftOut(MAX7219_DATA, MAX7219_CLOCK, MSBFIRST, address);
shiftOut(MAX7219_DATA, MAX7219_CLOCK, MSBFIRST, highByte(data));
// U2 now shifts the high byte into U6 while U2 gets the low byte (Lower 8 Displays Outer Segments)
shiftOut(MAX7219_DATA, MAX7219_CLOCK, MSBFIRST, address);
shiftOut(MAX7219_DATA, MAX7219_CLOCK, MSBFIRST, lowByte(data));
// Load U2 and U6
digitalWrite(MAX7219_LOAD, HIGH);
digitalWrite(MAX7219_LOAD, LOW);
digitalWrite(LdOuter, LOW);
}
// Note: the 'data' is 16 bits which are split into highByte (for U9) and lowByte (for (U5)
void InnerSegments(byte address, word data) {
digitalWrite(LdInner, HIGH); // AND Gate U1 sections B (clock signal) and C (load signal)
// U5 first gets the high byte (Upper 8 Displays Inner Segments) from the UNO
shiftOut(MAX7219_DATA, MAX7219_CLOCK, MSBFIRST, address);
shiftOut(MAX7219_DATA, MAX7219_CLOCK, MSBFIRST, highByte(data));
// U5 now shifts the high byte into U9 while U5 gets the low byte (Lower 8 Displays Inner Segments)
shiftOut(MAX7219_DATA, MAX7219_CLOCK, MSBFIRST, address);
shiftOut(MAX7219_DATA, MAX7219_CLOCK, MSBFIRST, lowByte(data));
// Load U5 and U9
digitalWrite(MAX7219_LOAD, HIGH);
digitalWrite(MAX7219_LOAD, LOW);
digitalWrite(LdInner, LOW);
}
// these are U2 and U6 Control Register Commands (data is 8 bits)
void CmdO(byte address, byte data) {
// AND Gate U1 sections A (clock signal) and D (load signal)
digitalWrite(LdOuter, HIGH);
// U2 first gets the address and data from the UNO
shiftOut(MAX7219_DATA, MAX7219_CLOCK, MSBFIRST, address);
shiftOut(MAX7219_DATA, MAX7219_CLOCK, MSBFIRST, data);
// U2 shifts into U6 (Note: U2 and U6 get the identical address and data)
shiftOut(MAX7219_DATA, MAX7219_CLOCK, MSBFIRST, address);
shiftOut(MAX7219_DATA, MAX7219_CLOCK, MSBFIRST, data);
// Load U2 and U6
digitalWrite(MAX7219_LOAD, HIGH);
digitalWrite(MAX7219_LOAD, LOW);
digitalWrite(LdOuter, LOW);
}
// these are U5 and U9 Control Register Commands (data is 8 bits)
void CmdI(byte address, byte data) {
// AND Gate U1 sections B (clock signal) and C (load signal)
digitalWrite(LdInner, HIGH);
// U5 first gets the address and data from the UNO
shiftOut(MAX7219_DATA, MAX7219_CLOCK, MSBFIRST, address);
shiftOut(MAX7219_DATA, MAX7219_CLOCK, MSBFIRST, data);
// U5 shifts into U9 (Note: U5 and U9 get the identical address and data)
shiftOut(MAX7219_DATA, MAX7219_CLOCK, MSBFIRST, address);
shiftOut(MAX7219_DATA, MAX7219_CLOCK, MSBFIRST, data);
// Load U5 and U9
digitalWrite(MAX7219_LOAD, HIGH);
digitalWrite(MAX7219_LOAD, LOW);
digitalWrite(LdInner, LOW);
}
// --------------------------------------------------------------------
void setup() {
// MAX7219 Control Register Set-Up
initialize();
// Outer Segment Set-Up (U2 and U6 - See MAX7219 Data Sheet page 7)
CmdO(0x0C, 0x00); // shutdown register
CmdO(0x0F, 0x00); // display test register - test mode off
CmdO(0x0B, 0x07); // scan limit register - display digits 0-7
CmdO(0x0A, 0x08); // intensity register - set display brightness
CmdO(0x09, 0x00); // decode mode register - set individual segments
CmdO(0x00, 0000); // No-Op
// Inner Segment Set-Up (U5 and U9 - See Max7219 Data Sheet page 7)
CmdI(0x0c, 0x00); // shutdown register
CmdI(0x0f, 0x00); // display test register - test mode off
CmdI(0x0b, 0x07); // scan limit register - display digits 0-7
CmdI(0x0a, 0x08); // intensity register - set display brightness
CmdI(0x09, 0x00); // decode mode register - set individual segments
CmdI(0x00, 0000); // No-Op
}
// -----------------------------------------------------------------------
void get_message(int msg) {
// get message string from progmem
strcpy_P(buffer, (char *)pgm_read_word(&(string_table[msg])));
String tempString = buffer;
char outSeg [MAX_SIZE];
char inSeg [MAX_SIZE];
for (int z = 0; z < 8; z++) {
int get_asciiHOO = tempString.charAt(z);
// Get the high order (Displays 9 thru 16) Outer Segment Codes from progmem
memcpy_P (&outSeg, &OSeg [get_asciiHOO - 32], sizeof outSeg);
char *Lo1;
int OuterHigh = strtol(outSeg, &Lo1, 2);
int get_asciiLOO = tempString.charAt(z + 8);
// Get the low order (Displays 1 thru 8) Outer Segment Codes from progmem
memcpy_P (&outSeg, &OSeg [get_asciiLOO - 32], sizeof outSeg);
char *Lo2;
int OuterLow = strtol(outSeg, &Lo2, 2);
// create the 16 bit word where 'high' is display 16, 'low' is display 8;
// each iteration thru this loop decrements until 'high'
// is display 9 and 'low' is display 1
word OUTERdata = 0;
OUTERdata = word(OuterHigh, OuterLow);
OuterSegments(8 - z, OUTERdata);
}
for (int z = 0; z < 8; z++) {
int get_asciiHOI = tempString.charAt(z);
// Get the high order (Displays 9 thru 16) Inner Segment Codes from progmem
memcpy_P (&inSeg, &ISeg[get_asciiHOI - 32], sizeof inSeg);
char *Li1;
int InnerHigh = strtol(inSeg, &Li1, 2);
int get_asciiLOI = tempString.charAt(z + 8);
// Get the low order (Displays 1 thru 8) Inner Segment Codes from progmem
memcpy_P(&inSeg, &ISeg[get_asciiLOI - 32], sizeof inSeg);
char *Li2;
int InnerLow = strtol(inSeg, &Li2, 2);
// create the 16 bit word where 'high' is display 16, 'low' is display 8;
// each iteration thru this loop decrements until 'high'
// is display 9 and 'low' is display 1
word INNERdata = 0;
INNERdata = word(InnerHigh, InnerLow);
InnerSegments(8 - z, INNERdata);
}
}
void loop() {
// alternate inner segment chips (U5 and U9) and outer segment chips (U2 and U6) shutdown modes
// the human eye cannot perceive the switch between inner and outer segments
CmdO(0x0c, 0001); // U2 and U6 exit Shutdown Mode
delay(10);
CmdO(0x0c, 0000); // U2 and U6 enter Shutdown Mode
CmdI(0x0c, 0001); // U5 and U9 exit Shutdown Mode
delay(10);
CmdI(0x0c, 0000); // U5 and U9 enter Shutdown Mode
if (millis() > time1 + Interval_Message) {
time1 = millis();
get_message(x);
x++;
if (x == 7) x = 0;
}
}
nano:12
nano:11
nano:10
nano:9
nano:8
nano:7
nano:6
nano:5
nano:4
nano:3
nano:2
nano:GND.2
nano:RESET.2
nano:0
nano:1
nano:13
nano:3.3V
nano:AREF
nano:A0
nano:A1
nano:A2
nano:A3
nano:A4
nano:A5
nano:A6
nano:A7
nano:5V
nano:RESET
nano:GND.1
nano:VIN
nano:12.2
nano:5V.2
nano:13.2
nano:11.2
nano:RESET.3
nano:GND.3
matrix1:V+
matrix1:GND
matrix1:DIN
matrix1:CS
matrix1:CLK
matrix1:V+.2
matrix1:GND.2
matrix1:DOUT
matrix1:CS.2
matrix1:CLK.2