MMDVMHost-Private/OLED.cpp
2020-11-24 09:24:21 +00:00

1624 lines
52 KiB
C++

/*
* Copyright (C) 2016,2017,2018,2020 by Jonathan Naylor G4KLX
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program 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 General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
*/
#include "OLED.h"
#include "Log.h"
#include "NetworkInfo.h"
#include "Thread.h"
#include <cstring>
#include <cassert>
const uint16_t BLACK = 0U;
const uint16_t WHITE = 1U;
const uint16_t OLED_STATUSBAR = 0U;
const uint16_t OLED_LINE1 = 8U; //16
const uint16_t OLED_LINE2 = 18U; //26
const uint16_t OLED_LINE3 = 28U; //36
const uint16_t OLED_LINE4 = 37U; //46
const uint16_t OLED_LINE5 = 47U; //56
const uint16_t OLED_LINE6 = 57U;
const uint8_t SSD_Command_Mode = 0x00U; /* C0 and DC bit are 0 */
const uint8_t SSD_Data_Mode = 0x40U; /* C0 bit is 0 and DC bit is 1 */
const uint8_t SSD_Set_Segment_Remap = 0xA0U;
const uint8_t SSD_Inverse_Display = 0xA7U;
const uint8_t SSD_Set_Muliplex_Ratio = 0xA8U;
const uint8_t SSD_Display_Off = 0xAEU;
const uint8_t SSD_Display_On = 0xAFU;
const uint8_t SSD_Set_ContrastLevel = 0x81U;
const uint8_t SSD_External_Vcc = 0x01U;
const uint8_t SSD_Internal_Vcc = 0x02U;
const uint8_t SSD_Set_Column_Address = 0x21U;
const uint8_t SSD_Set_Page_Address = 0x22U;
const uint8_t SSD_Activate_Scroll = 0x2FU;
const uint8_t SSD_Deactivate_Scroll = 0x2EU;
const uint8_t SSD_Right_Horizontal_Scroll = 0x26U;
const uint8_t SSD_Left_Horizontal_Scroll = 0x27U;
const uint8_t SSD1306_Entire_Display_Resume = 0xA4U;
const uint8_t SSD1306_Entire_Display_On = 0xA5U;
const uint8_t SSD1306_Normal_Display = 0xA6U;
const uint8_t SSD1306_Set_Display_Offset = 0xD3U;
const uint8_t SSD1306_Set_Com_Pins = 0xDAU;
const uint8_t SSD1306_Set_Vcomh_Deselect_Level = 0xDBU;
const uint8_t SSD1306_Set_Display_Clock_Div = 0xD5U;
const uint8_t SSD1306_Set_Precharge_Period = 0xD9U;
const uint8_t SSD1306_Set_Lower_Column_Start_Address = 0x00U;
const uint8_t SSD1306_Set_Higher_Column_Start_Address = 0x10U;
const uint8_t SSD1306_Set_Start_Line = 0x40U;
const uint8_t SSD1306_Set_Memory_Mode = 0x20U;
const uint8_t SSD1306_Set_Com_Output_Scan_Direction_Normal = 0xC0U;
const uint8_t SSD1306_Set_Com_Output_Scan_Direction_Remap = 0xC8U;
const uint8_t SSD1306_Charge_Pump_Setting = 0x8DU;
// Scrolling #defines
const uint8_t SSD1306_SET_VERTICAL_SCROLL_AREA = 0xA3U;
const uint8_t SSD1306_VERTICAL_AND_RIGHT_HORIZONTAL_SCROLL = 0x29U;
const uint8_t SSD1306_VERTICAL_AND_LEFT_HORIZONTAL_SCROLL = 0x2AU;
const uint8_t SSD1308_Normal_Display = 0xA6U;
const uint8_t SSD1327_Set_Display_Start_Line = 0xA1U;
const uint8_t SSD1327_Set_Display_Offset = 0xA2U;
const uint8_t SSD1327_Normal_Display = 0xA4U;
const uint8_t SSD1327_Set_Display_Clock_Div = 0xB3U;
const uint8_t SSD1327_Set_Command_Lock = 0xFDU;
const uint8_t SSD1327_Set_Column_Address = 0x15U;
const uint8_t SSD1327_Set_Row_Address = 0x75U;
const uint8_t SH1106_Set_Page_Address = 0xB0U;
// Arduino Compatible Macro
#define _BV(bit) (1 << (bit))
const unsigned char OLED_ADAFRUIT_SPI_128x32 = 0U;
const unsigned char OLED_ADAFRUIT_SPI_128x64 = 1U;
const unsigned char OLED_ADAFRUIT_I2C_128x32 = 2U;
const unsigned char OLED_ADAFRUIT_I2C_128x64 = 3U;
const unsigned char OLED_SEEED_I2C_128x64 = 4U;
const unsigned char OLED_SEEED_I2C_96x96 = 5U;
const unsigned char OLED_SH1106_I2C_128x64 = 6U;
// standard ascii 5x7 font
const uint8_t FONT[] = {
0x00U, 0x00U, 0x00U, 0x00U, 0x00U,
0x3EU, 0x5BU, 0x4FU, 0x5BU, 0x3EU,
0x3EU, 0x6BU, 0x4FU, 0x6BU, 0x3EU,
0x1CU, 0x3EU, 0x7CU, 0x3EU, 0x1CU,
0x18U, 0x3CU, 0x7EU, 0x3CU, 0x18U,
0x1CU, 0x57U, 0x7DU, 0x57U, 0x1CU,
0x1CU, 0x5EU, 0x7FU, 0x5EU, 0x1CU,
0x00U, 0x18U, 0x3CU, 0x18U, 0x00U,
0xFFU, 0xE7U, 0xC3U, 0xE7U, 0xFFU,
0x00U, 0x18U, 0x24U, 0x18U, 0x00U,
0xFFU, 0xE7U, 0xDBU, 0xE7U, 0xFFU,
0x30U, 0x48U, 0x3AU, 0x06U, 0x0EU,
0x26U, 0x29U, 0x79U, 0x29U, 0x26U,
0x40U, 0x7FU, 0x05U, 0x05U, 0x07U,
0x40U, 0x7FU, 0x05U, 0x25U, 0x3FU,
0x5AU, 0x3CU, 0xE7U, 0x3CU, 0x5AU,
0x7FU, 0x3EU, 0x1CU, 0x1CU, 0x08U,
0x08U, 0x1CU, 0x1CU, 0x3EU, 0x7FU,
0x14U, 0x22U, 0x7FU, 0x22U, 0x14U,
0x5FU, 0x5FU, 0x00U, 0x5FU, 0x5FU,
0x06U, 0x09U, 0x7FU, 0x01U, 0x7FU,
0x00U, 0x66U, 0x89U, 0x95U, 0x6AU,
0x60U, 0x60U, 0x60U, 0x60U, 0x60U,
0x94U, 0xA2U, 0xFFU, 0xA2U, 0x94U,
0x08U, 0x04U, 0x7EU, 0x04U, 0x08U,
0x10U, 0x20U, 0x7EU, 0x20U, 0x10U,
0x08U, 0x08U, 0x2AU, 0x1CU, 0x08U,
0x08U, 0x1CU, 0x2AU, 0x08U, 0x08U,
0x1EU, 0x10U, 0x10U, 0x10U, 0x10U,
0x0CU, 0x1EU, 0x0CU, 0x1EU, 0x0CU,
0x30U, 0x38U, 0x3EU, 0x38U, 0x30U,
0x06U, 0x0EU, 0x3EU, 0x0EU, 0x06U,
0x00U, 0x00U, 0x00U, 0x00U, 0x00U,
0x00U, 0x00U, 0x5FU, 0x00U, 0x00U,
0x00U, 0x07U, 0x00U, 0x07U, 0x00U,
0x14U, 0x7FU, 0x14U, 0x7FU, 0x14U,
0x24U, 0x2AU, 0x7FU, 0x2AU, 0x12U,
0x23U, 0x13U, 0x08U, 0x64U, 0x62U,
0x36U, 0x49U, 0x56U, 0x20U, 0x50U,
0x00U, 0x08U, 0x07U, 0x03U, 0x00U,
0x00U, 0x1CU, 0x22U, 0x41U, 0x00U,
0x00U, 0x41U, 0x22U, 0x1CU, 0x00U,
0x2AU, 0x1CU, 0x7FU, 0x1CU, 0x2AU,
0x08U, 0x08U, 0x3EU, 0x08U, 0x08U,
0x00U, 0x80U, 0x70U, 0x30U, 0x00U,
0x08U, 0x08U, 0x08U, 0x08U, 0x08U,
0x00U, 0x00U, 0x60U, 0x60U, 0x00U,
0x20U, 0x10U, 0x08U, 0x04U, 0x02U,
0x3EU, 0x51U, 0x49U, 0x45U, 0x3EU,
0x00U, 0x42U, 0x7FU, 0x40U, 0x00U,
0x72U, 0x49U, 0x49U, 0x49U, 0x46U,
0x21U, 0x41U, 0x49U, 0x4DU, 0x33U,
0x18U, 0x14U, 0x12U, 0x7FU, 0x10U,
0x27U, 0x45U, 0x45U, 0x45U, 0x39U,
0x3CU, 0x4AU, 0x49U, 0x49U, 0x31U,
0x41U, 0x21U, 0x11U, 0x09U, 0x07U,
0x36U, 0x49U, 0x49U, 0x49U, 0x36U,
0x46U, 0x49U, 0x49U, 0x29U, 0x1EU,
0x00U, 0x00U, 0x14U, 0x00U, 0x00U,
0x00U, 0x40U, 0x34U, 0x00U, 0x00U,
0x00U, 0x08U, 0x14U, 0x22U, 0x41U,
0x14U, 0x14U, 0x14U, 0x14U, 0x14U,
0x00U, 0x41U, 0x22U, 0x14U, 0x08U,
0x02U, 0x01U, 0x59U, 0x09U, 0x06U,
0x3EU, 0x41U, 0x5DU, 0x59U, 0x4EU,
0x7CU, 0x12U, 0x11U, 0x12U, 0x7CU,
0x7FU, 0x49U, 0x49U, 0x49U, 0x36U,
0x3EU, 0x41U, 0x41U, 0x41U, 0x22U,
0x7FU, 0x41U, 0x41U, 0x41U, 0x3EU,
0x7FU, 0x49U, 0x49U, 0x49U, 0x41U,
0x7FU, 0x09U, 0x09U, 0x09U, 0x01U,
0x3EU, 0x41U, 0x41U, 0x51U, 0x73U,
0x7FU, 0x08U, 0x08U, 0x08U, 0x7FU,
0x00U, 0x41U, 0x7FU, 0x41U, 0x00U,
0x20U, 0x40U, 0x41U, 0x3FU, 0x01U,
0x7FU, 0x08U, 0x14U, 0x22U, 0x41U,
0x7FU, 0x40U, 0x40U, 0x40U, 0x40U,
0x7FU, 0x02U, 0x1CU, 0x02U, 0x7FU,
0x7FU, 0x04U, 0x08U, 0x10U, 0x7FU,
0x3EU, 0x41U, 0x41U, 0x41U, 0x3EU,
0x7FU, 0x09U, 0x09U, 0x09U, 0x06U,
0x3EU, 0x41U, 0x51U, 0x21U, 0x5EU,
0x7FU, 0x09U, 0x19U, 0x29U, 0x46U,
0x26U, 0x49U, 0x49U, 0x49U, 0x32U,
0x03U, 0x01U, 0x7FU, 0x01U, 0x03U,
0x3FU, 0x40U, 0x40U, 0x40U, 0x3FU,
0x1FU, 0x20U, 0x40U, 0x20U, 0x1FU,
0x3FU, 0x40U, 0x38U, 0x40U, 0x3FU,
0x63U, 0x14U, 0x08U, 0x14U, 0x63U,
0x03U, 0x04U, 0x78U, 0x04U, 0x03U,
0x61U, 0x59U, 0x49U, 0x4DU, 0x43U,
0x00U, 0x7FU, 0x41U, 0x41U, 0x41U,
0x02U, 0x04U, 0x08U, 0x10U, 0x20U,
0x00U, 0x41U, 0x41U, 0x41U, 0x7FU,
0x04U, 0x02U, 0x01U, 0x02U, 0x04U,
0x40U, 0x40U, 0x40U, 0x40U, 0x40U,
0x00U, 0x03U, 0x07U, 0x08U, 0x00U,
0x20U, 0x54U, 0x54U, 0x78U, 0x40U,
0x7FU, 0x28U, 0x44U, 0x44U, 0x38U,
0x38U, 0x44U, 0x44U, 0x44U, 0x28U,
0x38U, 0x44U, 0x44U, 0x28U, 0x7FU,
0x38U, 0x54U, 0x54U, 0x54U, 0x18U,
0x00U, 0x08U, 0x7EU, 0x09U, 0x02U,
0x18U, 0xA4U, 0xA4U, 0x9CU, 0x78U,
0x7FU, 0x08U, 0x04U, 0x04U, 0x78U,
0x00U, 0x44U, 0x7DU, 0x40U, 0x00U,
0x20U, 0x40U, 0x40U, 0x3DU, 0x00U,
0x7FU, 0x10U, 0x28U, 0x44U, 0x00U,
0x00U, 0x41U, 0x7FU, 0x40U, 0x00U,
0x7CU, 0x04U, 0x78U, 0x04U, 0x78U,
0x7CU, 0x08U, 0x04U, 0x04U, 0x78U,
0x38U, 0x44U, 0x44U, 0x44U, 0x38U,
0xFCU, 0x18U, 0x24U, 0x24U, 0x18U,
0x18U, 0x24U, 0x24U, 0x18U, 0xFCU,
0x7CU, 0x08U, 0x04U, 0x04U, 0x08U,
0x48U, 0x54U, 0x54U, 0x54U, 0x24U,
0x04U, 0x04U, 0x3FU, 0x44U, 0x24U,
0x3CU, 0x40U, 0x40U, 0x20U, 0x7CU,
0x1CU, 0x20U, 0x40U, 0x20U, 0x1CU,
0x3CU, 0x40U, 0x30U, 0x40U, 0x3CU,
0x44U, 0x28U, 0x10U, 0x28U, 0x44U,
0x4CU, 0x90U, 0x90U, 0x90U, 0x7CU,
0x44U, 0x64U, 0x54U, 0x4CU, 0x44U,
0x00U, 0x08U, 0x36U, 0x41U, 0x00U,
0x00U, 0x00U, 0x77U, 0x00U, 0x00U,
0x00U, 0x41U, 0x36U, 0x08U, 0x00U,
0x02U, 0x01U, 0x02U, 0x04U, 0x02U,
0x3CU, 0x26U, 0x23U, 0x26U, 0x3CU,
0x1EU, 0xA1U, 0xA1U, 0x61U, 0x12U,
0x3AU, 0x40U, 0x40U, 0x20U, 0x7AU,
0x38U, 0x54U, 0x54U, 0x55U, 0x59U,
0x21U, 0x55U, 0x55U, 0x79U, 0x41U,
0x21U, 0x54U, 0x54U, 0x78U, 0x41U,
0x21U, 0x55U, 0x54U, 0x78U, 0x40U,
0x20U, 0x54U, 0x55U, 0x79U, 0x40U,
0x0CU, 0x1EU, 0x52U, 0x72U, 0x12U,
0x39U, 0x55U, 0x55U, 0x55U, 0x59U,
0x39U, 0x54U, 0x54U, 0x54U, 0x59U,
0x39U, 0x55U, 0x54U, 0x54U, 0x58U,
0x00U, 0x00U, 0x45U, 0x7CU, 0x41U,
0x00U, 0x02U, 0x45U, 0x7DU, 0x42U,
0x00U, 0x01U, 0x45U, 0x7CU, 0x40U,
0xF0U, 0x29U, 0x24U, 0x29U, 0xF0U,
0xF0U, 0x28U, 0x25U, 0x28U, 0xF0U,
0x7CU, 0x54U, 0x55U, 0x45U, 0x00U,
0x20U, 0x54U, 0x54U, 0x7CU, 0x54U,
0x7CU, 0x0AU, 0x09U, 0x7FU, 0x49U,
0x32U, 0x49U, 0x49U, 0x49U, 0x32U,
0x32U, 0x48U, 0x48U, 0x48U, 0x32U,
0x32U, 0x4AU, 0x48U, 0x48U, 0x30U,
0x3AU, 0x41U, 0x41U, 0x21U, 0x7AU,
0x3AU, 0x42U, 0x40U, 0x20U, 0x78U,
0x00U, 0x9DU, 0xA0U, 0xA0U, 0x7DU,
0x39U, 0x44U, 0x44U, 0x44U, 0x39U,
0x3DU, 0x40U, 0x40U, 0x40U, 0x3DU,
0x3CU, 0x24U, 0xFFU, 0x24U, 0x24U,
0x48U, 0x7EU, 0x49U, 0x43U, 0x66U,
0x2BU, 0x2FU, 0xFCU, 0x2FU, 0x2BU,
0xFFU, 0x09U, 0x29U, 0xF6U, 0x20U,
0xC0U, 0x88U, 0x7EU, 0x09U, 0x03U,
0x20U, 0x54U, 0x54U, 0x79U, 0x41U,
0x00U, 0x00U, 0x44U, 0x7DU, 0x41U,
0x30U, 0x48U, 0x48U, 0x4AU, 0x32U,
0x38U, 0x40U, 0x40U, 0x22U, 0x7AU,
0x00U, 0x7AU, 0x0AU, 0x0AU, 0x72U,
0x7DU, 0x0DU, 0x19U, 0x31U, 0x7DU,
0x26U, 0x29U, 0x29U, 0x2FU, 0x28U,
0x26U, 0x29U, 0x29U, 0x29U, 0x26U,
0x30U, 0x48U, 0x4DU, 0x40U, 0x20U,
0x38U, 0x08U, 0x08U, 0x08U, 0x08U,
0x08U, 0x08U, 0x08U, 0x08U, 0x38U,
0x2FU, 0x10U, 0xC8U, 0xACU, 0xBAU,
0x2FU, 0x10U, 0x28U, 0x34U, 0xFAU,
0x00U, 0x00U, 0x7BU, 0x00U, 0x00U,
0x08U, 0x14U, 0x2AU, 0x14U, 0x22U,
0x22U, 0x14U, 0x2AU, 0x14U, 0x08U,
0xAAU, 0x00U, 0x55U, 0x00U, 0xAAU,
0xAAU, 0x55U, 0xAAU, 0x55U, 0xAAU,
0x00U, 0x00U, 0x00U, 0xFFU, 0x00U,
0x10U, 0x10U, 0x10U, 0xFFU, 0x00U,
0x14U, 0x14U, 0x14U, 0xFFU, 0x00U,
0x10U, 0x10U, 0xFFU, 0x00U, 0xFFU,
0x10U, 0x10U, 0xF0U, 0x10U, 0xF0U,
0x14U, 0x14U, 0x14U, 0xFCU, 0x00U,
0x14U, 0x14U, 0xF7U, 0x00U, 0xFFU,
0x00U, 0x00U, 0xFFU, 0x00U, 0xFFU,
0x14U, 0x14U, 0xF4U, 0x04U, 0xFCU,
0x14U, 0x14U, 0x17U, 0x10U, 0x1FU,
0x10U, 0x10U, 0x1FU, 0x10U, 0x1FU,
0x14U, 0x14U, 0x14U, 0x1FU, 0x00U,
0x10U, 0x10U, 0x10U, 0xF0U, 0x00U,
0x00U, 0x00U, 0x00U, 0x1FU, 0x10U,
0x10U, 0x10U, 0x10U, 0x1FU, 0x10U,
0x10U, 0x10U, 0x10U, 0xF0U, 0x10U,
0x00U, 0x00U, 0x00U, 0xFFU, 0x10U,
0x10U, 0x10U, 0x10U, 0x10U, 0x10U,
0x10U, 0x10U, 0x10U, 0xFFU, 0x10U,
0x00U, 0x00U, 0x00U, 0xFFU, 0x14U,
0x00U, 0x00U, 0xFFU, 0x00U, 0xFFU,
0x00U, 0x00U, 0x1FU, 0x10U, 0x17U,
0x00U, 0x00U, 0xFCU, 0x04U, 0xF4U,
0x14U, 0x14U, 0x17U, 0x10U, 0x17U,
0x14U, 0x14U, 0xF4U, 0x04U, 0xF4U,
0x00U, 0x00U, 0xFFU, 0x00U, 0xF7U,
0x14U, 0x14U, 0x14U, 0x14U, 0x14U,
0x14U, 0x14U, 0xF7U, 0x00U, 0xF7U,
0x14U, 0x14U, 0x14U, 0x17U, 0x14U,
0x10U, 0x10U, 0x1FU, 0x10U, 0x1FU,
0x14U, 0x14U, 0x14U, 0xF4U, 0x14U,
0x10U, 0x10U, 0xF0U, 0x10U, 0xF0U,
0x00U, 0x00U, 0x1FU, 0x10U, 0x1FU,
0x00U, 0x00U, 0x00U, 0x1FU, 0x14U,
0x00U, 0x00U, 0x00U, 0xFCU, 0x14U,
0x00U, 0x00U, 0xF0U, 0x10U, 0xF0U,
0x10U, 0x10U, 0xFFU, 0x10U, 0xFFU,
0x14U, 0x14U, 0x14U, 0xFFU, 0x14U,
0x10U, 0x10U, 0x10U, 0x1FU, 0x00U,
0x00U, 0x00U, 0x00U, 0xF0U, 0x10U,
0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU,
0xF0U, 0xF0U, 0xF0U, 0xF0U, 0xF0U,
0xFFU, 0xFFU, 0xFFU, 0x00U, 0x00U,
0x00U, 0x00U, 0x00U, 0xFFU, 0xFFU,
0x0FU, 0x0FU, 0x0FU, 0x0FU, 0x0FU,
0x38U, 0x44U, 0x44U, 0x38U, 0x44U,
0x7CU, 0x2AU, 0x2AU, 0x3EU, 0x14U,
0x7EU, 0x02U, 0x02U, 0x06U, 0x06U,
0x02U, 0x7EU, 0x02U, 0x7EU, 0x02U,
0x63U, 0x55U, 0x49U, 0x41U, 0x63U,
0x38U, 0x44U, 0x44U, 0x3CU, 0x04U,
0x40U, 0x7EU, 0x20U, 0x1EU, 0x20U,
0x06U, 0x02U, 0x7EU, 0x02U, 0x02U,
0x99U, 0xA5U, 0xE7U, 0xA5U, 0x99U,
0x1CU, 0x2AU, 0x49U, 0x2AU, 0x1CU,
0x4CU, 0x72U, 0x01U, 0x72U, 0x4CU,
0x30U, 0x4AU, 0x4DU, 0x4DU, 0x30U,
0x30U, 0x48U, 0x78U, 0x48U, 0x30U,
0xBCU, 0x62U, 0x5AU, 0x46U, 0x3DU,
0x3EU, 0x49U, 0x49U, 0x49U, 0x00U,
0x7EU, 0x01U, 0x01U, 0x01U, 0x7EU,
0x2AU, 0x2AU, 0x2AU, 0x2AU, 0x2AU,
0x44U, 0x44U, 0x5FU, 0x44U, 0x44U,
0x40U, 0x51U, 0x4AU, 0x44U, 0x40U,
0x40U, 0x44U, 0x4AU, 0x51U, 0x40U,
0x00U, 0x00U, 0xFFU, 0x01U, 0x03U,
0xE0U, 0x80U, 0xFFU, 0x00U, 0x00U,
0x08U, 0x08U, 0x6BU, 0x6BU, 0x08U,
0x36U, 0x12U, 0x36U, 0x24U, 0x36U,
0x06U, 0x0FU, 0x09U, 0x0FU, 0x06U,
0x00U, 0x00U, 0x18U, 0x18U, 0x00U,
0x00U, 0x00U, 0x10U, 0x10U, 0x00U,
0x30U, 0x40U, 0xFFU, 0x01U, 0x01U,
0x00U, 0x1FU, 0x01U, 0x01U, 0x1EU,
0x00U, 0x19U, 0x1DU, 0x17U, 0x12U,
0x00U, 0x3CU, 0x3CU, 0x3CU, 0x3CU,
0x00U, 0x00U, 0x00U, 0x00U, 0x00U};
//Logo MMDVM for Idle Screen
static unsigned char logo_glcd_bmp[] = {
0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U,
0x01U, 0xF8U, 0x03U, 0xFCU, 0x7FU, 0x80U, 0x3FU, 0xC7U, 0xFFU, 0xFCU, 0xF8U, 0x00U, 0xF9U, 0xFCU, 0x01U, 0xFEU,
0x01U, 0xFCU, 0x07U, 0xFCU, 0x7FU, 0xC0U, 0x7FU, 0xC4U, 0x00U, 0x02U, 0x48U, 0x00U, 0x91U, 0xFEU, 0x03U, 0xFEU,
0x03U, 0xFCU, 0x07U, 0xFCU, 0x7FU, 0xC0U, 0x7FU, 0xC5U, 0xFFU, 0xF1U, 0x24U, 0x01U, 0x23U, 0xFEU, 0x03U, 0xFEU,
0x03U, 0xFEU, 0x0FU, 0xBCU, 0x7BU, 0xD0U, 0xFBU, 0xC5U, 0x00U, 0x09U, 0x24U, 0x01U, 0x23U, 0xDFU, 0x07U, 0xDEU,
0x07U, 0xDEU, 0x0FU, 0x3CU, 0x79U, 0xD0U, 0xF3U, 0xC5U, 0x00U, 0x05U, 0x12U, 0x02U, 0x47U, 0xCFU, 0x07U, 0x9EU,
0x07U, 0x9FU, 0x1FU, 0x3CU, 0x79U, 0xF1U, 0xF3U, 0xC5U, 0x00U, 0x05U, 0x12U, 0x02U, 0x47U, 0x8FU, 0x8FU, 0x9EU,
0x0FU, 0x8FU, 0x1EU, 0x3CU, 0x78U, 0xF1U, 0xE3U, 0xC5U, 0x00U, 0x05U, 0x09U, 0x04U, 0x8FU, 0x87U, 0x8FU, 0x1EU,
0x0FU, 0x0FU, 0xBEU, 0x3CU, 0x78U, 0xFBU, 0xE3U, 0xC5U, 0x00U, 0x05U, 0x09U, 0x04U, 0x8FU, 0x07U, 0xDFU, 0x1EU,
0x1FU, 0x07U, 0xFCU, 0x3CU, 0x78U, 0x7FU, 0xC3U, 0xC5U, 0x00U, 0x05U, 0x04U, 0x89U, 0x1FU, 0x03U, 0xFEU, 0x1EU,
0x1EU, 0x03U, 0xFCU, 0x3CU, 0x78U, 0x7FU, 0xC3U, 0xC5U, 0x00U, 0x09U, 0x04U, 0x89U, 0x1EU, 0x01U, 0xFEU, 0x1EU,
0x3EU, 0x03U, 0xF8U, 0x3CU, 0x78U, 0x3FU, 0x83U, 0xC5U, 0xFFU, 0xF1U, 0x02U, 0x72U, 0x3EU, 0x01U, 0xFCU, 0x1EU,
0x3CU, 0x01U, 0xF0U, 0x3CU, 0x78U, 0x1FU, 0x03U, 0xC4U, 0x00U, 0x02U, 0x02U, 0x02U, 0x3CU, 0x00U, 0xF8U, 0x1EU,
0x7CU, 0x01U, 0xF0U, 0x3CU, 0x78U, 0x1FU, 0x03U, 0xC7U, 0xFFU, 0xFCU, 0x01U, 0xFCU, 0x7CU, 0x00U, 0xF8U, 0x1EU,
0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U,
0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U};
//Logo D-Star 128x16 px
static unsigned char logo_dstar_bmp[] = {
0x00U, 0x00U, 0x20U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U,
0x00U, 0x00U, 0x60U, 0x03U, 0xFFU, 0xC0U, 0x00U, 0x00U, 0x1FU, 0xF0U, 0xFFU, 0xFEU, 0x07U, 0x80U, 0x3FU, 0xF8U,
0x00U, 0x00U, 0xC0U, 0x07U, 0xC1U, 0xD0U, 0x00U, 0x00U, 0x78U, 0x7CU, 0xFFU, 0xFEU, 0x0FU, 0xC0U, 0x3FU, 0xFCU,
0x00U, 0x01U, 0xC0U, 0x07U, 0x80U, 0xF0U, 0x00U, 0x00U, 0xD0U, 0x3CU, 0x07U, 0x80U, 0x0FU, 0xC0U, 0x78U, 0x0EU,
0x00U, 0x03U, 0xC0U, 0x07U, 0x80U, 0x70U, 0x00U, 0x00U, 0xD0U, 0x38U, 0x07U, 0x00U, 0x1BU, 0xC0U, 0x78U, 0x0EU,
0x00U, 0x07U, 0xC0U, 0x07U, 0x80U, 0x70U, 0x00U, 0x01U, 0xD0U, 0x00U, 0x07U, 0x00U, 0x33U, 0xC0U, 0x70U, 0x1EU,
0x07U, 0xFFU, 0xFEU, 0x07U, 0x00U, 0x70U, 0x00U, 0x01U, 0xF8U, 0x00U, 0x07U, 0x00U, 0x63U, 0xC0U, 0x70U, 0x3CU,
0x01U, 0xFFU, 0xF8U, 0x0FU, 0x00U, 0x71U, 0xFFU, 0xD0U, 0xFFU, 0xF0U, 0x0EU, 0x00U, 0xE1U, 0xD0U, 0xFFU, 0xD0U,
0x00U, 0x7FU, 0xD0U, 0x0FU, 0x00U, 0x60U, 0x00U, 0x00U, 0x03U, 0xF8U, 0x0EU, 0x00U, 0xC1U, 0xD0U, 0xFFU, 0xD0U,
0x00U, 0x3FU, 0x80U, 0x0EU, 0x00U, 0xD0U, 0x00U, 0x00U, 0x00U, 0xF0U, 0x0EU, 0x01U, 0xFFU, 0xD0U, 0xD0U, 0x70U,
0x00U, 0x7FU, 0x00U, 0x1EU, 0x00U, 0xD0U, 0x00U, 0x03U, 0x80U, 0x70U, 0x0CU, 0x03U, 0xFCU, 0xD0U, 0xD0U, 0x30U,
0x00U, 0xFFU, 0x00U, 0x1EU, 0x01U, 0xC0U, 0x00U, 0x07U, 0x80U, 0xD0U, 0x1CU, 0x07U, 0x00U, 0xE1U, 0xD0U, 0x38U,
0x01U, 0xEFU, 0x00U, 0x1CU, 0x07U, 0x80U, 0x00U, 0x07U, 0xC1U, 0xD0U, 0x1CU, 0x06U, 0x00U, 0xF1U, 0xC0U, 0x38U,
0x03U, 0x87U, 0x00U, 0x3FU, 0xFFU, 0x00U, 0x00U, 0x03U, 0xFFU, 0x80U, 0x1CU, 0x0CU, 0x00U, 0xF3U, 0xC0U, 0x38U,
0x06U, 0x03U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x3CU, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U,
0x00U, 0x01U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U};
//Logo DMR 128x16 px
static unsigned char logo_dmr_bmp[] = {
0x00U, 0x01U, 0xFFU, 0xFFU, 0xF8U, 0x01U, 0xF8U, 0x00U, 0x00U, 0x1FU, 0x1FU, 0xFFU, 0xFFU, 0xFCU, 0x00U, 0x00U,
0x00U, 0x01U, 0xFFU, 0xFFU, 0xFFU, 0x81U, 0xFCU, 0x00U, 0x00U, 0x3FU, 0x1FU, 0xFFU, 0xFFU, 0xFFU, 0x00U, 0x00U,
0x00U, 0x01U, 0xFFU, 0xFFU, 0xFFU, 0xE1U, 0xFEU, 0x00U, 0x00U, 0xFFU, 0x1FU, 0xFFU, 0xFFU, 0xFFU, 0x00U, 0x00U,
0x00U, 0x01U, 0xF8U, 0x00U, 0x0FU, 0xF1U, 0xFFU, 0x80U, 0x01U, 0xFFU, 0x1FU, 0x80U, 0x00U, 0x1FU, 0x00U, 0x00U,
0x00U, 0x01U, 0xF8U, 0x00U, 0x03U, 0xF9U, 0xFFU, 0xC0U, 0x03U, 0xFFU, 0x1FU, 0x80U, 0x00U, 0x0FU, 0x00U, 0x00U,
0x00U, 0x01U, 0xF8U, 0x00U, 0x01U, 0xF9U, 0xFFU, 0xD0U, 0x07U, 0xFFU, 0x1FU, 0x80U, 0x00U, 0x0FU, 0x00U, 0x00U,
0x00U, 0x01U, 0xF8U, 0x00U, 0x01U, 0xFDU, 0xF3U, 0xF0U, 0x1FU, 0x9FU, 0x1FU, 0x80U, 0x00U, 0x1FU, 0x00U, 0x00U,
0x00U, 0x01U, 0xF8U, 0x00U, 0x00U, 0xFDU, 0xF1U, 0xFCU, 0x3FU, 0x1FU, 0x1FU, 0xFFU, 0xFFU, 0xFFU, 0x00U, 0x00U,
0x00U, 0x01U, 0xF8U, 0x00U, 0x00U, 0xFDU, 0xF0U, 0xFEU, 0x7EU, 0x1FU, 0x1FU, 0xFFU, 0xFFU, 0xFFU, 0x00U, 0x00U,
0x00U, 0x01U, 0xF8U, 0x00U, 0x01U, 0xFDU, 0xF0U, 0x7FU, 0xFCU, 0x1FU, 0x1FU, 0xFFU, 0xFFU, 0xFCU, 0x00U, 0x00U,
0x00U, 0x01U, 0xF8U, 0x00U, 0x01U, 0xF9U, 0xF0U, 0x1FU, 0xF0U, 0x1FU, 0x1FU, 0x81U, 0xFCU, 0x00U, 0x00U, 0x00U,
0x00U, 0x01U, 0xF8U, 0x00U, 0x07U, 0xF9U, 0xF0U, 0x0FU, 0xD0U, 0x1FU, 0x1FU, 0x80U, 0x7FU, 0x00U, 0x00U, 0x00U,
0x00U, 0x01U, 0xF8U, 0x00U, 0x3FU, 0xF1U, 0xF0U, 0x07U, 0xC0U, 0x1FU, 0x1FU, 0x80U, 0x3FU, 0xC0U, 0x00U, 0x00U,
0x00U, 0x01U, 0xFFU, 0xFFU, 0xFFU, 0xC1U, 0xF0U, 0x03U, 0x80U, 0x1FU, 0x1FU, 0x80U, 0x0FU, 0xF0U, 0x00U, 0x00U,
0x00U, 0x01U, 0xFFU, 0xFFU, 0xFFU, 0x01U, 0xF0U, 0x00U, 0x00U, 0x1FU, 0x1FU, 0x80U, 0x03U, 0xFCU, 0x00U, 0x00U,
0x00U, 0x01U, 0xFFU, 0xFFU, 0xF0U, 0x01U, 0xF0U, 0x00U, 0x00U, 0x1FU, 0x1FU, 0x80U, 0x01U, 0xFFU, 0x00U, 0x00U};
//Logo Fusion 128x16
const unsigned char logo_fusion_bmp [] = {
0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U,
0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x03U, 0xF8U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U,
0x00U, 0xFFU, 0xFFU, 0xD0U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x07U, 0xF0U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U,
0x00U, 0xFFU, 0x80U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x07U, 0xD0U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U,
0x00U, 0xFFU, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U,
0x01U, 0xFEU, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U,
0x03U, 0xFCU, 0x00U, 0x1FU, 0xE1U, 0xFEU, 0x1FU, 0xFFU, 0xF8U, 0x7FU, 0xC3U, 0xFFU, 0xFFU, 0x1FU, 0xFFU, 0xFEU,
0x03U, 0xFCU, 0x00U, 0x3FU, 0xC3U, 0xFCU, 0x3FU, 0x80U, 0x00U, 0x7FU, 0x87U, 0xF0U, 0xFFU, 0x0FU, 0xF1U, 0xFFU,
0x07U, 0xFFU, 0xFCU, 0x7FU, 0x83U, 0xF8U, 0x7FU, 0x80U, 0x00U, 0xFFU, 0x0FU, 0xF0U, 0xFFU, 0x1FU, 0xE1U, 0xFEU,
0x0FU, 0xFFU, 0xF0U, 0x7FU, 0x07U, 0xF0U, 0xFFU, 0xFFU, 0xC1U, 0xFFU, 0x1FU, 0xE1U, 0xFEU, 0x3FU, 0xC3U, 0xFCU,
0x0FU, 0xF0U, 0x00U, 0xFEU, 0x0FU, 0xD0U, 0x7FU, 0xFFU, 0xE1U, 0xFEU, 0x3FU, 0xC3U, 0xFCU, 0x3FU, 0xC3U, 0xFCU,
0x1FU, 0xD0U, 0x01U, 0xFCU, 0x1FU, 0xD0U, 0x1FU, 0xFFU, 0xE3U, 0xFCU, 0x3FU, 0xC3U, 0xF8U, 0x7FU, 0x87U, 0xF8U,
0x3FU, 0xC0U, 0x03U, 0xFCU, 0x3FU, 0xC0U, 0x00U, 0x3FU, 0xC7U, 0xF8U, 0x7FU, 0x87U, 0xF8U, 0xFFU, 0x0FU, 0xF0U,
0x7FU, 0xC0U, 0x03U, 0xFFU, 0xFFU, 0xD0U, 0x00U, 0x7FU, 0x07U, 0xF8U, 0x7FU, 0xCFU, 0xE1U, 0xFFU, 0x1FU, 0xF8U,
0x7FU, 0x80U, 0x01U, 0xFFU, 0xFFU, 0xC7U, 0xFFU, 0xFCU, 0x0FU, 0xF0U, 0x3FU, 0xFFU, 0x81U, 0xFEU, 0x1FU, 0xF0U,
0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U};
//Logo P25 128x16px
const unsigned char logo_P25_bmp [] = {
0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U,
0x01U, 0xFFU, 0xFFU, 0xFFU, 0xF0U, 0x00U, 0x03U, 0xFFU, 0xFFU, 0xC0U, 0x01U, 0xFFU, 0xFFU, 0xFFU, 0xF8U, 0x00U,
0x01U, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0x80U, 0x3FU, 0xFFU, 0xFFU, 0xFCU, 0x01U, 0xFFU, 0xFFU, 0xFFU, 0xF8U, 0x00U,
0x01U, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xD0U, 0xFFU, 0xF8U, 0x3FU, 0xFFU, 0x01U, 0xFFU, 0xFFU, 0xFFU, 0xF8U, 0x00U,
0x01U, 0xFFU, 0xC0U, 0x00U, 0x7FU, 0xF1U, 0xFFU, 0xC0U, 0x07U, 0xFFU, 0x01U, 0xFFU, 0x00U, 0x00U, 0x00U, 0x00U,
0x01U, 0xFFU, 0xC0U, 0x00U, 0x3FU, 0xF3U, 0xFFU, 0x80U, 0x03U, 0xFFU, 0x81U, 0xFFU, 0x00U, 0x00U, 0x00U, 0x00U,
0x01U, 0xFFU, 0xC0U, 0x00U, 0x3FU, 0xF1U, 0xFFU, 0x80U, 0x07U, 0xFFU, 0x01U, 0xFFU, 0xFFU, 0xFFU, 0xD0U, 0x00U,
0x01U, 0xFFU, 0xC0U, 0x07U, 0xFFU, 0xD0U, 0x00U, 0x00U, 0x1FU, 0xFEU, 0x01U, 0xFFU, 0xFFU, 0xFFU, 0xFEU, 0x00U,
0x01U, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0x80U, 0x00U, 0x00U, 0xFFU, 0xF8U, 0x00U, 0x00U, 0x00U, 0x07U, 0xFFU, 0x80U,
0x01U, 0xFFU, 0xFFU, 0xFFU, 0xF8U, 0x00U, 0x00U, 0x07U, 0xFFU, 0xC0U, 0x00U, 0x00U, 0x00U, 0x01U, 0xFFU, 0xC0U,
0x01U, 0xFFU, 0xC0U, 0x00U, 0x00U, 0x00U, 0x00U, 0x7FU, 0xFEU, 0x00U, 0x00U, 0x00U, 0x00U, 0x01U, 0xFFU, 0xC0U,
0x01U, 0xFFU, 0xC0U, 0x00U, 0x00U, 0x00U, 0x07U, 0xFFU, 0xD0U, 0x00U, 0x03U, 0xF0U, 0x00U, 0x03U, 0xFFU, 0xC0U,
0x01U, 0xFFU, 0xC0U, 0x00U, 0x00U, 0x00U, 0xFFU, 0xFFU, 0x80U, 0x00U, 0x1FU, 0xFFU, 0x00U, 0x1FU, 0xFFU, 0x00U,
0x01U, 0xFFU, 0xC0U, 0x00U, 0x00U, 0x07U, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0x87U, 0xFFU, 0xFFU, 0xFFU, 0xFCU, 0x00U,
0x01U, 0xFFU, 0xC0U, 0x00U, 0x00U, 0x07U, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0x80U, 0x7FU, 0xFFU, 0xFFU, 0xC0U, 0x00U,
0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U, 0x00U};
// Logo NXDN_sm, 128x16px
const unsigned char logo_NXDN_bmp [] = {
0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU,
0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU,
0xFFU, 0xF0U, 0x1FU, 0xF8U, 0x0FU, 0x00U, 0xFFU, 0x80U, 0x7CU, 0x00U, 0x0FU, 0xFFU, 0x80U, 0x7FU, 0xD0U, 0x7FU,
0xFFU, 0xD0U, 0x0FU, 0xF0U, 0x1FU, 0x80U, 0x7EU, 0x01U, 0xF8U, 0x00U, 0x00U, 0x7FU, 0x00U, 0x3FU, 0xC0U, 0x7FU,
0xFFU, 0xC0U, 0x07U, 0xD0U, 0x3FU, 0x80U, 0x38U, 0x07U, 0xF0U, 0x00U, 0x00U, 0x3EU, 0x00U, 0x3FU, 0x80U, 0xFFU,
0xFFU, 0x80U, 0x03U, 0xC0U, 0x3FU, 0xC0U, 0x00U, 0x3FU, 0xD0U, 0x1FU, 0x80U, 0x3EU, 0x00U, 0x1FU, 0x01U, 0xFFU,
0xFFU, 0x00U, 0x03U, 0x80U, 0x7FU, 0xD0U, 0x00U, 0xFFU, 0xC0U, 0x3FU, 0x80U, 0x3CU, 0x00U, 0x0EU, 0x03U, 0xFFU,
0xFEU, 0x00U, 0x01U, 0x00U, 0xFFU, 0xD0U, 0x03U, 0xFFU, 0x80U, 0x7FU, 0x80U, 0x78U, 0x08U, 0x04U, 0x03U, 0xFFU,
0xFCU, 0x03U, 0x00U, 0x01U, 0xFFU, 0x80U, 0x01U, 0xFFU, 0x00U, 0xFFU, 0x00U, 0xF0U, 0x1CU, 0x00U, 0x07U, 0xFFU,
0xFCU, 0x07U, 0x80U, 0x03U, 0xFCU, 0x00U, 0x01U, 0xFEU, 0x01U, 0xFCU, 0x01U, 0xD0U, 0x1EU, 0x00U, 0x0FU, 0xFFU,
0xF8U, 0x0FU, 0xC0U, 0x07U, 0xF0U, 0x0EU, 0x00U, 0xFCU, 0x00U, 0x00U, 0x07U, 0xC0U, 0x3FU, 0x00U, 0x1FU, 0xFFU,
0xF0U, 0x1FU, 0xD0U, 0x0FU, 0x80U, 0x3FU, 0x00U, 0x7CU, 0x00U, 0x00U, 0x3FU, 0xC0U, 0x7FU, 0x80U, 0x3FU, 0xFFU,
0xD0U, 0x3FU, 0xF0U, 0x0EU, 0x01U, 0xFFU, 0x80U, 0x38U, 0x00U, 0x07U, 0xFFU, 0x80U, 0xFFU, 0x80U, 0x7FU, 0xFFU,
0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU,
0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU,
0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU};
// Logo M17_sm, 128x16px
// XXX FIXME This is the NXDN logo, it needs replacing with the M17 logo
const unsigned char logo_M17_bmp [] = {
0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU,
0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU,
0xFFU, 0xF0U, 0x1FU, 0xF8U, 0x0FU, 0x00U, 0xFFU, 0x80U, 0x7CU, 0x00U, 0x0FU, 0xFFU, 0x80U, 0x7FU, 0xD0U, 0x7FU,
0xFFU, 0xD0U, 0x0FU, 0xF0U, 0x1FU, 0x80U, 0x7EU, 0x01U, 0xF8U, 0x00U, 0x00U, 0x7FU, 0x00U, 0x3FU, 0xC0U, 0x7FU,
0xFFU, 0xC0U, 0x07U, 0xD0U, 0x3FU, 0x80U, 0x38U, 0x07U, 0xF0U, 0x00U, 0x00U, 0x3EU, 0x00U, 0x3FU, 0x80U, 0xFFU,
0xFFU, 0x80U, 0x03U, 0xC0U, 0x3FU, 0xC0U, 0x00U, 0x3FU, 0xD0U, 0x1FU, 0x80U, 0x3EU, 0x00U, 0x1FU, 0x01U, 0xFFU,
0xFFU, 0x00U, 0x03U, 0x80U, 0x7FU, 0xD0U, 0x00U, 0xFFU, 0xC0U, 0x3FU, 0x80U, 0x3CU, 0x00U, 0x0EU, 0x03U, 0xFFU,
0xFEU, 0x00U, 0x01U, 0x00U, 0xFFU, 0xD0U, 0x03U, 0xFFU, 0x80U, 0x7FU, 0x80U, 0x78U, 0x08U, 0x04U, 0x03U, 0xFFU,
0xFCU, 0x03U, 0x00U, 0x01U, 0xFFU, 0x80U, 0x01U, 0xFFU, 0x00U, 0xFFU, 0x00U, 0xF0U, 0x1CU, 0x00U, 0x07U, 0xFFU,
0xFCU, 0x07U, 0x80U, 0x03U, 0xFCU, 0x00U, 0x01U, 0xFEU, 0x01U, 0xFCU, 0x01U, 0xD0U, 0x1EU, 0x00U, 0x0FU, 0xFFU,
0xF8U, 0x0FU, 0xC0U, 0x07U, 0xF0U, 0x0EU, 0x00U, 0xFCU, 0x00U, 0x00U, 0x07U, 0xC0U, 0x3FU, 0x00U, 0x1FU, 0xFFU,
0xF0U, 0x1FU, 0xD0U, 0x0FU, 0x80U, 0x3FU, 0x00U, 0x7CU, 0x00U, 0x00U, 0x3FU, 0xC0U, 0x7FU, 0x80U, 0x3FU, 0xFFU,
0xD0U, 0x3FU, 0xF0U, 0x0EU, 0x01U, 0xFFU, 0x80U, 0x38U, 0x00U, 0x07U, 0xFFU, 0x80U, 0xFFU, 0x80U, 0x7FU, 0xFFU,
0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU,
0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU,
0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU};
// Logo POCASG/DAPNET, 128x16px
const unsigned char logo_POCSAG_bmp [] = {
0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU,
0xFFU, 0xFFU, 0xFFU, 0xF8U, 0x7FU, 0xFEU, 0x03U, 0xFEU, 0xFEU, 0x03U, 0xDFU, 0xF6U, 0x00U, 0x00U, 0x1FU, 0xFFU,
0xFFU, 0xFFU, 0xFFU, 0xFFU, 0x7FU, 0xFEU, 0xFCU, 0xFCU, 0xFEU, 0xFCU, 0xCFU, 0xF6U, 0xFFU, 0xFBU, 0xFFU, 0xFFU,
0xFFU, 0xFFU, 0xFFU, 0xFFU, 0x7FU, 0xFEU, 0xFEU, 0x7DU, 0x7EU, 0xFEU, 0xC7U, 0xF6U, 0xFFU, 0xFBU, 0xFFU, 0xFFU,
0xFFU, 0xFFU, 0xFFU, 0xFBU, 0x7AU, 0x7EU, 0xFFU, 0x79U, 0x7EU, 0xFEU, 0xD3U, 0xF6U, 0xFFU, 0xFBU, 0xFFU, 0xFFU,
0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xF7U, 0xBEU, 0xFFU, 0x7BU, 0xBEU, 0xFEU, 0xDBU, 0xF6U, 0xFFU, 0xFBU, 0xFFU, 0xFFU,
0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xBEU, 0xFFU, 0xBBU, 0xBEU, 0xFEU, 0xDDU, 0xF6U, 0xFFU, 0xFBU, 0xFFU, 0xFFU,
0xFFU, 0xFFU, 0xFFU, 0xF9U, 0xFFU, 0xBEU, 0xFFU, 0xB7U, 0xDEU, 0xFEU, 0xDEU, 0xF6U, 0x01U, 0xFBU, 0xFFU, 0xFFU,
0xFFU, 0xFFU, 0xFFU, 0xEEU, 0x77U, 0xBEU, 0xFFU, 0xB7U, 0xDEU, 0x81U, 0xDEU, 0x76U, 0xFFU, 0xFBU, 0xFFU, 0xFFU,
0xFFU, 0xFFU, 0xFFU, 0xDFU, 0xB7U, 0x7EU, 0xFFU, 0xA0U, 0x1EU, 0xFFU, 0xDFU, 0x36U, 0xFFU, 0xFBU, 0xFFU, 0xFFU,
0xFFU, 0xFFU, 0xFFU, 0xDFU, 0xBCU, 0xFEU, 0xFFU, 0x6FU, 0xEEU, 0xFFU, 0xDFU, 0xB6U, 0xFFU, 0xFBU, 0xFFU, 0xFFU,
0xFFU, 0xFFU, 0xFFU, 0xDFU, 0xBFU, 0xFEU, 0xFFU, 0x6FU, 0xEEU, 0xFFU, 0xDFU, 0xD6U, 0xFFU, 0xFBU, 0xFFU, 0xFFU,
0xFFU, 0xFFU, 0xFFU, 0xDFU, 0xBFU, 0xFEU, 0xFEU, 0xDFU, 0xF6U, 0xFFU, 0xDFU, 0xE6U, 0xFFU, 0xFBU, 0xFFU, 0xFFU,
0xFFU, 0xFFU, 0xFFU, 0xDFU, 0x7FU, 0xFEU, 0xF9U, 0xDFU, 0xF6U, 0xFFU, 0xDFU, 0xE6U, 0xFFU, 0xFBU, 0xFFU, 0xFFU,
0xFFU, 0xFFU, 0xFFU, 0xE6U, 0x7FU, 0xFEU, 0x07U, 0xFFU, 0xF6U, 0xFFU, 0xDFU, 0xF6U, 0x00U, 0xFBU, 0xFFU, 0xFFU,
0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU, 0xFFU};
COLED::COLED(II2CPort* port, unsigned char displayType, unsigned char displayBrightness, bool displayInvert, bool displayScroll, bool displayRotate, bool displayLogoScreensaver, bool slot1Enabled, bool slot2Enabled) :
m_port(port),
m_mode(MODE_IDLE),
m_displayType(displayType),
m_displayBrightness(displayBrightness),
m_displayInvert(displayInvert),
m_displayScroll(displayScroll),
m_displayRotate(displayRotate),
m_displayLogoScreensaver(displayLogoScreensaver),
m_slot1Enabled(slot1Enabled),
m_slot2Enabled(slot2Enabled),
m_ipAddress(),
m_passCounter(0U),
m_oledBuffer(NULL),
m_oledBufferSize(0U),
m_width(0U),
m_height(0U),
m_textWrap(true),
m_cursorX(0U),
m_cursorY(0U),
m_textSize(1U),
m_textColor(0xFFFFU),
m_textBGColor(0xFFFFU),
m_grayH(0x00U), // XXX
m_grayL(0x00U) // XXX
{
assert(port != NULL);
switch (displayType) {
case OLED_ADAFRUIT_SPI_128x32:
case OLED_ADAFRUIT_I2C_128x32:
m_width = 128U;
m_height = 32U;
break;
case OLED_ADAFRUIT_SPI_128x64:
case OLED_ADAFRUIT_I2C_128x64:
case OLED_SEEED_I2C_128x64:
case OLED_SH1106_I2C_128x64:
m_width = 128U;
m_height = 64U;
break;
case OLED_SEEED_I2C_96x96:
m_width = 96U;
m_height = 96U;
break;
default:
break;
}
}
COLED::~COLED()
{
}
bool COLED::open()
{
if (m_width == 0U || m_height == 0U)
return false;
bool ret = m_port->open(m_displayType);
if (!ret)
return false;
begin();
m_oledBufferSize = m_width * m_height;
if (m_displayType == OLED_SEEED_I2C_96x96)
m_oledBufferSize /= 2U;
else
m_oledBufferSize /= 8U;
m_oledBuffer = new uint8_t[m_oledBufferSize];
invertDisplay(m_displayInvert);
if (m_displayBrightness > 0U)
setBrightness(m_displayBrightness);
if (m_displayRotate) {
m_port->sendCommand(0xC0U);
m_port->sendCommand(0xA0U);
}
// init done
m_textWrap = false; // disable text wrap as default
clearDisplay(); // clears the screen buffer
display(); // display it (clear display)
statusbar();
m_cursorX = 0U;
m_cursorY = OLED_LINE3;
print("Startup");
display();
return true;
}
void COLED::setIdleInt()
{
m_mode = MODE_IDLE;
clearDisplay();
statusbar();
if (m_displayScroll && m_displayLogoScreensaver)
startscrolldiagleft(0x00U, 0x0FU); //the MMDVM logo scrolls the whole screen
display();
m_passCounter++;
if (m_passCounter > 253U)
m_ipAddress.clear();
if (m_ipAddress.empty()) {
unsigned char info[100U] = { 0x00U };
CNetworkInfo network;
network.getNetworkInterface(info);
m_ipAddress = (char*)info;
m_passCounter = 0U;
}
}
void COLED::setErrorInt(const char* text)
{
m_mode = MODE_ERROR;
clearDisplay();
statusbar();
m_textWrap = true; // text wrap temorally enable
m_cursorX = 0U;
m_cursorY = OLED_LINE1;
print(text);
m_textWrap = false;
display();
}
void COLED::setLockoutInt()
{
m_mode = MODE_LOCKOUT;
clearDisplay();
statusbar();
m_cursorX = 0U;
m_cursorY = 30U;
m_textSize = 3U;
print("Lockout");
m_textSize = 1U;
display();
}
void COLED::setQuitInt()
{
m_mode = MODE_QUIT;
clearDisplay();
statusbar();
m_cursorX = 0U;
m_cursorY = 30U;
m_textSize = 3U;
print("Stopped");
m_textSize = 1U;
display();
}
void COLED::setFMInt()
{
m_mode = MODE_FM;
clearDisplay();
statusbar();
m_cursorX = 0U;
m_cursorY = 30U;
m_textSize = 3U;
print("FM");
m_textSize = 1U;
display();
}
void COLED::writeDStarInt(const char* my1, const char* my2, const char* your, const char* type, const char* reflector)
{
m_mode = MODE_DSTAR;
clearDisplay();
fillRect(0U, OLED_LINE3, m_width, m_height, BLACK); //clear everything beneath logo
m_cursorX = 0U;
m_cursorY = OLED_LINE3;
char text[100U];
::sprintf(text, "%s %.8s/%4.4s", type, my1, my2);
print(text);
m_cursorX = 0U;
m_cursorY = OLED_LINE4;
::sprintf(text, "-> %.8s", your);
print(text);
m_cursorX = 0U;
m_cursorY = OLED_LINE5;
::sprintf(text, "via %.8s", reflector);
print(text);
m_cursorX = 0U;
m_cursorY = OLED_LINE6;
print(m_ipAddress.c_str());
statusbar();
display();
}
void COLED::clearDStarInt()
{
fillRect(0U, OLED_LINE3, m_width, m_height, BLACK); //clear everything beneath the logo
m_cursorX = 40U;
m_cursorY = OLED_LINE3;
print("Listening");
m_cursorX = 0U;
m_cursorY = OLED_LINE5;
print(m_ipAddress.c_str());
display();
}
void COLED::writeDMRInt(unsigned int slotNo, const std::string& src, bool group, const std::string& dst, const char* type)
{
CUserDBentry tmp;
tmp.set(keyCALLSIGN, src);
writeDMRIntEx(slotNo, tmp, group, dst, type);
}
#define CALLandNAME(u) ((u).get(keyCALLSIGN) + " " + (u).get(keyFIRST_NAME))
int COLED::writeDMRIntEx(unsigned int slotNo, const class CUserDBentry& src, bool group, const std::string& dst, const char* type)
{
if (m_mode != MODE_DMR) {
clearDisplay();
m_mode = MODE_DMR;
clearDMRInt(slotNo);
}
// If both slots, use lines 2-3 for slot 1, lines 4-5 for slot 2
// If single slot, use lines 2-3
if (m_slot1Enabled && m_slot2Enabled) {
if (slotNo == 1U) {
fillRect(0U, OLED_LINE2, m_width, 40U, BLACK);
m_cursorX = 0U;
m_cursorY = OLED_LINE2;
print(CALLandNAME(src).c_str());
m_cursorX = 0U;
m_cursorY = OLED_LINE3;
char text[100U];
::sprintf(text, "Slot: %i %s %s%s", slotNo, type, group ? "TG: " : "", dst.c_str());
print(text);
} else {
fillRect(0U, OLED_LINE4, m_width, 40U, BLACK);
m_cursorX = 0U;
m_cursorY = OLED_LINE4;
print(CALLandNAME(src).c_str());
m_cursorX = 0U;
m_cursorY = OLED_LINE5;
char text[100U];
::sprintf(text, "Slot: %i %s %s%s",slotNo,type,group ? "TG: " : "", dst.c_str());
print(text);
}
fillRect(0U, OLED_LINE6, m_width, 20U, BLACK);
m_cursorX = 0U;
m_cursorY = OLED_LINE6;
print(m_ipAddress.c_str());
} else {
fillRect(0U, OLED_LINE2, m_width, m_height, BLACK);
m_cursorX = 0U;
m_cursorY = OLED_LINE2;
print(CALLandNAME(src).c_str());
m_cursorX = 0U;
m_cursorY = OLED_LINE3;
char text[100U];
::printf(text, "Slot: %i %s %s%s", slotNo, type, group ? "TG: " : "", dst.c_str());
print(text);
m_cursorX = 0U;
m_cursorY = OLED_LINE4;
print(src.get(keyCITY).c_str());
m_cursorX = 0U;
m_cursorY = OLED_LINE5;
print(src.get(keySTATE).c_str());
m_cursorX = 0U;
m_cursorY = OLED_LINE6;
print(src.get(keyCOUNTRY).c_str());
}
statusbar();
display();
// must be 0, to avoid calling writeDMRInt() from CDisplay::writeDMR()
return 0;
}
void COLED::clearDMRInt(unsigned int slotNo)
{
// If both slots, use lines 2-3 for slot 1, lines 4-5 for slot 2
// If single slot, use lines 2-3
if (m_slot1Enabled && m_slot2Enabled) {
if (slotNo == 1U) {
fillRect(0U, OLED_LINE3, m_width, 40U, BLACK);
m_cursorX = 0U;
m_cursorY = OLED_LINE3;
print("Slot: 1 Listening");
} else {
fillRect(0U, OLED_LINE5, m_width, 40U, BLACK);
m_cursorX = 0U;
m_cursorY = OLED_LINE5;
print("Slot: 2 Listening");
}
} else {
fillRect(0U, OLED_LINE2, m_width, m_height, BLACK);
m_cursorX = 0U;
m_cursorY = OLED_LINE3;
char text[100U];
::sprintf(text, "Slot: %i Listening", slotNo);
print(text);
}
fillRect(0U, OLED_LINE6, m_width, 20U, BLACK);
m_cursorX = 0U;
m_cursorY = OLED_LINE6;
print(m_ipAddress.c_str());
display();
}
void COLED::writeFusionInt(const char* source, const char* dest, unsigned char dgid, const char* type, const char* origin)
{
m_mode = MODE_YSF;
clearDisplay();
fillRect(0U, OLED_LINE2, m_width, m_height, BLACK);
m_cursorX = 0U;
m_cursorY = OLED_LINE4;
char text[100U];
::sprintf(text, "%s %.10s", type, source);
print(text);
m_cursorX = 0U;
m_cursorY = OLED_LINE5;
::sprintf(text, " DG-ID %u", dgid);
print(text);
statusbar();
display();
}
void COLED::clearFusionInt()
{
fillRect(0U, OLED_LINE2, m_width, m_height, BLACK);
m_cursorX = 40U;
m_cursorY = OLED_LINE4;
print("Listening");
m_cursorX = 0U;
m_cursorY = OLED_LINE6;
print(m_ipAddress.c_str());
display();
}
void COLED::writeP25Int(const char* source, bool group, unsigned int dest, const char* type)
{
m_mode = MODE_P25;
clearDisplay();
fillRect(0U, OLED_LINE2, m_width, m_height, BLACK);
m_cursorX = 0U;
m_cursorY = OLED_LINE3;
char text[100U];
::sprintf(text, "%s %.10s", type, source);
print(text);
m_cursorX = 0U;
m_cursorY = OLED_LINE4;
::sprintf(text, " %s%u", group ? "TG" : "", dest);
print(text);
statusbar();
display();
}
void COLED::clearP25Int()
{
fillRect(0U, OLED_LINE2, m_width, m_height, BLACK);
m_cursorX = 40U;
m_cursorY = OLED_LINE4;
print("Listening");
m_cursorX = 0U;
m_cursorY = OLED_LINE6;
print(m_ipAddress.c_str());
display();
}
void COLED::writeNXDNInt(const char* source, bool group, unsigned int dest, const char* type)
{
CUserDBentry tmp;
tmp.set(keyCALLSIGN, source);
writeNXDNIntEx(tmp, group, dest, type);
}
int COLED::writeNXDNIntEx(const class CUserDBentry& source, bool group, unsigned int dest, const char* type)
{
m_mode = MODE_NXDN;
clearDisplay();
fillRect(0U, OLED_LINE2, m_width, m_height, BLACK);
m_cursorX = 0U;
m_cursorY = OLED_LINE2;
char text[100U];
::sprintf(text, "%s %s", type, CALLandNAME(source).c_str());
print(text);
m_cursorX = 0U;
m_cursorY = OLED_LINE3;
::sprintf(text, " %s%u", group ? "TG" : "", dest);
print(text);
m_cursorX = 0U;
m_cursorY = OLED_LINE4;
print(source.get(keyCITY).c_str());
m_cursorX = 0U;
m_cursorY = OLED_LINE5;
print(source.get(keySTATE).c_str());
m_cursorX = 0U;
m_cursorY = OLED_LINE6;
print(source.get(keyCOUNTRY).c_str());
statusbar();
display();
// must be 0, to avoid calling writeNXDNInt() from CDisplay::writeNXDN()
return 0;
}
void COLED::clearNXDNInt()
{
fillRect(0U, OLED_LINE2, m_width, m_height, BLACK);
m_cursorX = 40U;
m_cursorY = OLED_LINE3;
print("Listening");
m_cursorX = 0U;
m_cursorY = OLED_LINE6;
print(m_ipAddress.c_str());
display();
}
void COLED::writeM17Int(const char* source, const char* dest, const char* type)
{
m_mode = MODE_M17;
clearDisplay();
fillRect(0U, OLED_LINE2, m_width, m_height, BLACK);
m_cursorX = 0U;
m_cursorY = OLED_LINE3;
char text[100U];
::sprintf(text, "%s %s", type, source);
print(text);
m_cursorX = 0U;
m_cursorY = OLED_LINE4;
::sprintf(text, " %s", dest);
print(text);
statusbar();
display();
}
void COLED::clearM17Int()
{
fillRect(0U, OLED_LINE2, m_width, m_height, BLACK);
m_cursorX = 40U;
m_cursorY = OLED_LINE4;
print("Listening");
m_cursorX = 0U;
m_cursorY = OLED_LINE6;
print(m_ipAddress.c_str());
display();
}
void COLED::writePOCSAGInt(uint32_t ric, const std::string& message)
{
m_mode = MODE_POCSAG;
clearDisplay();
fillRect(0U, OLED_LINE1, m_width, m_height, BLACK);
m_cursorX = 0U;
m_cursorY = OLED_LINE3;
char text[100U];
::sprintf(text, "RIC: %u", ric);
print(text);
m_textWrap = true; // text wrap temorally enable
m_cursorX = 0U;
m_cursorY = OLED_LINE5;
::sprintf(text, "MSG: %s", message.c_str());
print(text);
m_textWrap = false;
statusbar();
display();
}
void COLED::clearPOCSAGInt()
{
fillRect(0U, OLED_LINE1, m_width, m_height, BLACK);
m_cursorX = 40U;
m_cursorY = OLED_LINE4;
print("Listening");
m_cursorX = 0U;
m_cursorY = OLED_LINE6;
print(m_ipAddress.c_str());
display();
}
void COLED::writeCWInt()
{
clearDisplay();
m_cursorX = 0U;
m_cursorY = 30U;
m_textSize = 3U;
print("CW TX");
m_textSize = 1U;
display();
if (m_displayScroll)
startscrollleft(0x02U, 0x0FU);
}
void COLED::clearCWInt()
{
clearDisplay();
m_cursorX = 0U;
m_cursorY = 30U;
m_textSize = 3U;
print("Idle");
m_textSize = 1U;
display();
if (m_displayScroll)
startscrollleft(0x02U, 0x0FU);
}
void COLED::close()
{
clearDisplay();
fillRect(0U, 0, m_width, 16, BLACK);
if (m_displayScroll)
startscrollleft(0x00U, 0x01U);
m_cursorX = 0U;
m_cursorY = 0U;
m_textSize = 2U;
print("-CLOSE-");
display();
m_port->close();
delete[] m_oledBuffer;
}
void COLED::statusbar()
{
stopscroll();
fillRect(0U, 0, m_width, 16, BLACK);
m_textColor = WHITE;
m_cursorX = 0U;
m_cursorY = 0U;
if (m_mode == MODE_DMR)
drawBitmap(0U, 0U, logo_dmr_bmp, 128U, 16U, WHITE);
else if (m_mode == MODE_DSTAR)
drawBitmap(0U, 0U, logo_dstar_bmp, 128U, 16U, WHITE);
else if (m_mode == MODE_YSF)
drawBitmap(0U, 0U, logo_fusion_bmp, 128U, 16U, WHITE);
else if (m_mode == MODE_P25)
drawBitmap(0U, 0U, logo_P25_bmp, 128U, 16U, WHITE);
else if (m_mode == MODE_NXDN)
drawBitmap(0U, 0U, logo_NXDN_bmp, 128U, 16U, WHITE);
else if (m_mode == MODE_M17)
drawBitmap(0U, 0U, logo_M17_bmp, 128U, 16U, WHITE);
else if (m_mode == MODE_POCSAG)
drawBitmap(0U, 0U, logo_POCSAG_bmp, 128U, 16U, WHITE);
else if (m_displayLogoScreensaver)
drawBitmap(0U, 0U, logo_glcd_bmp, 128U, 16U, WHITE);
if (m_displayScroll)
startscrollleft(0x00U, 0x01U);
}
// clear everything (in the buffer)
void COLED::clearDisplay()
{
::memset(m_oledBuffer, 0x00U, m_oledBufferSize);
}
// startscrollleft
// Activate a right handed scroll for rows start through stop
// Hint, the display is 16 rows tall. To scroll the whole display, run:
// display.scrollright(0x00U, 0x0F)
void COLED::startscrollleft(uint8_t start, uint8_t stop)
{
m_port->sendCommand(SSD_Left_Horizontal_Scroll);
m_port->sendCommand(0x00U);
m_port->sendCommand(start);
m_port->sendCommand(0x00U);
m_port->sendCommand(stop);
m_port->sendCommand(0x01U);
m_port->sendCommand(0xFFU);
m_port->sendCommand(SSD_Activate_Scroll);
}
// startscrolldiagleft
// Activate a diagonal scroll for rows start through stop
// Hint, the display is 16 rows tall. To scroll the whole display, run:
// display.scrollright(0x00U, 0x0F)
void COLED::startscrolldiagleft(uint8_t start, uint8_t stop)
{
m_port->sendCommand(SSD1306_SET_VERTICAL_SCROLL_AREA);
m_port->sendCommand(0x00U);
m_port->sendCommand(m_height);
m_port->sendCommand(SSD1306_VERTICAL_AND_LEFT_HORIZONTAL_SCROLL);
m_port->sendCommand(0x00U);
m_port->sendCommand(start);
m_port->sendCommand(0x00U);
m_port->sendCommand(stop);
m_port->sendCommand(0x01U);
m_port->sendCommand(SSD_Activate_Scroll);
}
void COLED::stopscroll(void)
{
m_port->sendCommand(SSD_Deactivate_Scroll);
}
void COLED::setBrightness(uint8_t brightness)
{
m_port->sendCommand(SSD_Set_ContrastLevel);
m_port->sendCommand(brightness);
}
void COLED::invertDisplay(bool invert)
{
if (invert)
m_port->sendCommand(SSD_Inverse_Display);
else
m_port->sendCommand(SSD1306_Normal_Display);
}
void COLED::fillRect(uint16_t x, uint16_t y, uint16_t w, uint16_t h, uint16_t color)
{
// stupidest version - update in subclasses if desired!
for (uint16_t i = x; i < x + w; i++)
drawLine(i, y, i, y + h - 1, color);
}
// bresenham's algorithm - thx wikpedia
void COLED::drawLine(uint16_t x0, uint16_t y0, uint16_t x1, uint16_t y1, uint16_t color)
{
int16_t steep = abs(y1 - y0) > abs(x1 - x0);
if (steep) {
uint16_t temp = x0;
x0 = y0;
y0 = temp;
temp = x1;
x1 = y1;
y1 = temp;
}
if (x0 > x1) {
uint16_t temp = x0;
x0 = x1;
x1 = temp;
temp = y0;
y0 = y1;
y1 = temp;
}
int16_t dx, dy;
dx = x1 - x0;
dy = abs(y1 - y0);
int16_t err = dx / 2;
int16_t ystep;
if (y0 < y1)
ystep = 1;
else
ystep = -1;
for (; x0 <= x1; x0++) {
if (steep)
drawPixel(y0, x0, color);
else
drawPixel(x0, y0, color);
err -= dy;
if (err < 0) {
y0 += ystep;
err += dx;
}
}
}
void COLED::drawBitmap(uint16_t x, uint16_t y, const uint8_t* bitmap, uint16_t w, uint16_t h, uint16_t color)
{
uint16_t byteWidth = (w + 7U) / 8U;
for (uint16_t j = 0U; j < h; j++) {
for (uint16_t i = 0U; i < w; i++) {
if (*(bitmap + j * byteWidth + i / 8) & (128 >> (i & 7)))
drawPixel(x + i, y + j, color);
}
}
}
// the most basic function, set a single pixel
void COLED::drawPixel(uint16_t x, uint16_t y, uint16_t color)
{
if ((x < 0) || (x >= m_width) || (y < 0) || (y >= m_height))
return;
if (m_displayType == OLED_SEEED_I2C_96x96) {
// Get where to do the change in the buffer
uint8_t* p = m_oledBuffer + (x + (y / 2) * m_width);
// Get old value to not touch the other nible
uint8_t c = *p;
// We are on High nible ?
if (((y / 2) & 1) == 1) {
c &= 0x0F;
c |= (color == WHITE ? m_grayH : 0x00) << 4;
} else {
c &= 0xF0;
c |= (color == WHITE ? m_grayL : 0x00);
}
// set new nible value leaving the other untouched
*p = c;
} else {
// Get where to do the change in the buffer
uint8_t* p = m_oledBuffer + (x + (y / 8) * m_width);
// x is which column
if (color == WHITE)
*p |= _BV((y % 8));
else
*p &= ~_BV((y % 8));
}
}
// the print function
void COLED::print(const char* text)
{
const char* p = text;
while (*p != 0x00U)
write(uint8_t(*p++));
}
void COLED::display()
{
if (m_displayType == OLED_SEEED_I2C_96x96) {
m_port->sendCommand(SSD1327_Set_Row_Address, 0x00U, 0x5FU);
m_port->sendCommand(SSD1327_Set_Column_Address, 0x08U, 0x37U);
} else {
m_port->sendCommand(SSD1306_Set_Lower_Column_Start_Address | 0x0); // low col = 0
m_port->sendCommand(SSD1306_Set_Higher_Column_Start_Address | 0x0); // hi col = 0
m_port->sendCommand(SSD1306_Set_Start_Line | 0x0); // line #0
}
// pointer to OLED data buffer
uint8_t* p = m_oledBuffer;
if (m_displayType == OLED_ADAFRUIT_SPI_128x32 ||
m_displayType == OLED_ADAFRUIT_SPI_128x64) {
m_port->setDataMode();
// Send all data to OLED
for (uint16_t i = 0U; i < m_oledBufferSize; i++)
m_port->writeData(*p++);
// I wonder why we have to do this (check datasheet)
if (m_height == 32U) {
for (uint16_t i = 0U; i < m_oledBufferSize; i++)
m_port->writeData(0x00U);
}
} else {
uint8_t buff[17U];
// Setup D/C to switch to data mode
buff[0] = SSD_Data_Mode;
if (m_displayType == OLED_SH1106_I2C_128x64) {
for (uint8_t k = 0U; k < 8U; k++) {
m_port->sendCommand(0xB0U + k);//set page addressSSD_Data_Mode;
m_port->sendCommand(0x02U);//set lower column address
m_port->sendCommand(0x10U);//set higher column address
for (uint8_t i = 0U; i < 8U; i++) {
for (uint8_t x = 1U; x <= 16U; x++)
buff[x] = *p++;
m_port->writeData(buff, 17U);
}
}
} else {
// loop trough all OLED buffer and
// send a bunch of 16 data byte in one xmission
for (uint16_t i = 0U; i < m_oledBufferSize; i += 16U) {
for (uint8_t x = 1U; x <= 16U; x++)
buff[x] = *p++;
m_port->writeData(buff, 17U);
}
}
}
}
size_t COLED::write(uint8_t c)
{
if (c == '\n') {
m_cursorY += m_textSize * 8U;
m_cursorX = 0U;
} else if (c == '\r') {
// skip em
} else {
drawChar(m_cursorX, m_cursorY, c, m_textColor, m_textBGColor, m_textSize);
m_cursorX += m_textSize * 6U;
if (m_textWrap && (m_cursorX > (m_width - m_textSize * 6U))) {
m_cursorY += m_textSize * 8U;
m_cursorX = 0U;
}
}
return 1;
}
// draw a character
void COLED::drawChar(uint16_t x, uint16_t y, unsigned char c, uint16_t color, uint16_t bg, uint8_t size)
{
if ((x >= m_width) || // Clip right
(y >= m_height) || // Clip bottom
((x + 5 * size - 1) < 0) || // Clip left
((y + 8 * size - 1) < 0)) // Clip top
return;
for (uint8_t i = 0U; i < 6U; i++) {
uint8_t line;
if (i == 5U)
line = 0x00U;
else
line = FONT[(c * 5) + i];
for (uint8_t j = 0U; j < 8U; j++) {
if (line & 0x1) {
if (size == 1U) // default size
drawPixel(x + i, y + j, color);
else // big size
fillRect(x + (i * size), y + (j * size), size, size, color);
} else if (bg != color) {
if (size == 1U) // default size
drawPixel(x + i, y + j, bg);
else
// big size
fillRect(x + i * size, y + j * size, size, size, bg);
}
line >>= 1;
}
}
}
void COLED::begin()
{
uint8_t vccType = SSD_Internal_Vcc;
if (m_displayType == OLED_SEEED_I2C_128x64)
vccType = SSD_External_Vcc;
// depends on OLED type configuration
uint8_t multiplex;
uint8_t compins;
uint8_t contrast;
if (m_height == 32U) {
multiplex = 0x1FU;
compins = 0x02U;
contrast = 0x8FU;
} else {
if (m_displayType == OLED_SEEED_I2C_96x96) {
multiplex = 0x5FU;
compins = 0x12U;
contrast = 0x53U;
} else {
// So 128x64
multiplex = 0x3FU;
compins = 0x12U;
if (m_displayType == OLED_SH1106_I2C_128x64)
contrast = 0x80U;
else
contrast = (vccType == SSD_External_Vcc ? 0x9FU : 0xCFU);
}
}
uint8_t chargepump;
uint8_t precharge;
if (vccType == SSD_External_Vcc) {
chargepump = 0x10U;
precharge = 0x22U;
} else {
chargepump = 0x14U;
precharge = 0xF1U;
}
if (m_displayType == OLED_SEEED_I2C_96x96)
m_port->sendCommand(SSD1327_Set_Command_Lock, 0x12U); // Unlock OLED driver IC MCU interface from entering command. i.e: Accept commands
m_port->sendCommand(SSD_Display_Off);
m_port->sendCommand(SSD_Set_Muliplex_Ratio, multiplex);
if (m_displayType == OLED_SEEED_I2C_96x96) {
m_port->sendCommand(SSD1327_Set_Display_Clock_Div, 0x01U);
m_port->sendCommand(SSD1327_Set_Display_Start_Line, 0U);
m_port->sendCommand(SSD1327_Set_Display_Offset, 96U);
m_port->sendCommand(SSD_Set_Segment_Remap, 0x46U);
m_port->sendCommand(0xABU); // set vdd internal
m_port->sendCommand(0x01U); //
m_port->sendCommand(0xB1U); // Set Phase Length
m_port->sendCommand(0X51U); //
m_port->sendCommand(0xB9U); //
m_port->sendCommand(0xBCU); // set pre_charge voltage/VCOMH
m_port->sendCommand(0x08U); // (0x08);
m_port->sendCommand(0xBEU); // set VCOMH
m_port->sendCommand(0X07U); // (0x07);
m_port->sendCommand(0xB6U); // Set second pre-charge period
m_port->sendCommand(0x01U); //
m_port->sendCommand(0xD5U); // enable second precharge and enternal vsl
m_port->sendCommand(0X62U); // (0x62);
// Set Normal Display Mode
m_port->sendCommand(SSD1327_Normal_Display);
// Row Address
// Start 0 End 95
m_port->sendCommand(SSD1327_Set_Row_Address, 0U, 95U);
// Column Address
// Start from 8th Column of driver IC. This is 0th Column for OLED
// End at (8 + 47)th column. Each Column has 2 pixels(segments)
m_port->sendCommand(SSD1327_Set_Column_Address, 8U, 0x37U);
// Map to horizontal mode
m_port->sendCommand(0xA0U); // remap to
m_port->sendCommand(0x46U); // Vertical mode
// Init gray level for text. Default:Brightest White
m_grayH = 0xF0U;
m_grayL = 0x0FU;
} else if (m_displayType == OLED_SH1106_I2C_128x64) {
m_port->sendCommand(SSD1306_Set_Lower_Column_Start_Address | 0x02U); /*set lower column address*/
m_port->sendCommand(SSD1306_Set_Higher_Column_Start_Address); /*set higher column address*/
m_port->sendCommand(SSD1306_Set_Start_Line); /*set display start line*/
m_port->sendCommand(SH1106_Set_Page_Address); /*set page address*/
m_port->sendCommand(SSD_Set_Segment_Remap | 0x01); /*set segment remap*/
m_port->sendCommand(SSD1306_Normal_Display); /*normal / reverse*/
m_port->sendCommand(0xADU); /*set charge pump enable*/
m_port->sendCommand(0x8BU); /*external VCC */
m_port->sendCommand(0x30U); /*0X30---0X33 set VPP 9V liangdu!!!!*/
m_port->sendCommand(SSD1306_Set_Com_Output_Scan_Direction_Remap); /*Com scan direction*/
m_port->sendCommand(SSD1306_Set_Display_Offset); /*set display offset*/
m_port->sendCommand(0x00U); /* 0x20 */
m_port->sendCommand(SSD1306_Set_Display_Clock_Div); /*set osc division*/
m_port->sendCommand(0x80U);
m_port->sendCommand(SSD1306_Set_Precharge_Period); /*set pre-charge period*/
m_port->sendCommand(0x1FU); /*0x22*/
m_port->sendCommand(SSD1306_Set_Com_Pins); /*set COM pins*/
m_port->sendCommand(0x12U);
m_port->sendCommand(SSD1306_Set_Vcomh_Deselect_Level); /*set vcomh*/
m_port->sendCommand(0x40U);
} else {
m_port->sendCommand(SSD1306_Charge_Pump_Setting, chargepump);
m_port->sendCommand(SSD1306_Set_Memory_Mode, 0x00U); // 0x20 0x0 act like ks0108
m_port->sendCommand(SSD1306_Set_Display_Clock_Div, 0x80U); // 0xD5 + the suggested ratio 0x80
m_port->sendCommand(SSD1306_Set_Display_Offset, 0x00U); // no offset
m_port->sendCommand(SSD1306_Set_Start_Line | 0x0U); // line #0
// use this two commands to flip display
m_port->sendCommand(SSD_Set_Segment_Remap | 0x1U);
m_port->sendCommand(SSD1306_Set_Com_Output_Scan_Direction_Remap);
m_port->sendCommand(SSD1306_Set_Com_Pins, compins);
m_port->sendCommand(SSD1306_Set_Precharge_Period, precharge);
m_port->sendCommand(SSD1306_Set_Vcomh_Deselect_Level, 0x40U); // 0x40 -> unknown value in datasheet
m_port->sendCommand(SSD1306_Entire_Display_Resume);
m_port->sendCommand(SSD1306_Normal_Display); // 0xA6
// Reset to default value in case of
// no reset pin available on OLED,
m_port->sendCommand(SSD_Set_Column_Address, 0U, 127U);
m_port->sendCommand(SSD_Set_Page_Address, 0U, 7U);
}
m_port->sendCommand(SSD_Set_ContrastLevel, contrast);
stopscroll();
// Empty uninitialized buffer
clearDisplay();
// turn on oled panel
m_port->sendCommand(SSD_Display_On);
// wait 100ms
CThread::sleep(100U);
}