/* * This 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 3 of the License, or * (at your option) any later version. * * DogLcd 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. * * You should have received a copy of the GNU Lesser General Public License * along with DogLcd. If not, see . * * Copyright 2010 Eberhard Fahle */ #include "DogLcd.h" #include #include #include #if (ARDUINO >= 100) #include #else #include #endif DogLcd::DogLcd(int lcdSI, int lcdCLK, int lcdRS, int lcdCSB, int lcdRESET, int backLight) { this->lcdSI=lcdSI; this->lcdCLK=lcdCLK; this->lcdRS=lcdRS; this->lcdCSB=lcdCSB; this->lcdRESET=lcdRESET; this->backLight=backLight; } int DogLcd::begin(int model, int contrast, int vcc) { //init all pins to go HIGH, we dont want to send any commands by accident pinMode(this->lcdCSB,OUTPUT); digitalWrite(this->lcdCSB,HIGH); pinMode(this->lcdSI,OUTPUT); digitalWrite(this->lcdSI,HIGH); pinMode(this->lcdCLK,OUTPUT); digitalWrite(this->lcdCLK,HIGH); pinMode(this->lcdRS,OUTPUT); digitalWrite(this->lcdRS,HIGH); if(this->lcdRESET!=-1) { pinMode(this->lcdRESET,OUTPUT); digitalWrite(this->lcdRESET,HIGH); } if(this->backLight!=-1) { pinMode(this->backLight,OUTPUT); digitalWrite(this->backLight,LOW); } if(model==DOG_LCD_M081) { this->model=model; rows=1; cols=8; memSize=80; startAddress[0]=0; startAddress[1]=-1; startAddress[2]=-1; } else if(model==DOG_LCD_M162) { this->model=model; rows=2; cols=16; memSize=40; startAddress[0]=0; startAddress[1]=0x40; startAddress[2]=-1; } else if(model==DOG_LCD_M163) { this->model=model; rows=3; cols=16; memSize=16; startAddress[0]=0; startAddress[1]=0x10; startAddress[2]=0x20; } else { //unknown or unsupported model return -1; } if(contrast < 0 || contrast> 0x3F) { //contrast is outside the valid range return -1; } this->contrast=contrast; if(vcc==DOG_LCD_VCC_5V || vcc==DOG_LCD_VCC_3V3) { this->vcc=vcc; } else { //unknown or unsupported supply voltage return -1; } reset(); return 0; } void DogLcd::reset() { if(lcdRESET!=-1) { //If user wired the reset line, pull it low and wait for 40 millis digitalWrite(lcdRESET,LOW); delay(40); digitalWrite(lcdRESET,HIGH); delay(40); } else { //User wants software reset, we simply wait a bit for stable power delay(50); } if(model==DOG_LCD_M081) { //8-bit,1-line instructionSetTemplate=(uint8_t)0x30; } else if(model==DOG_LCD_M162) { //8-bit,2-line instructionSetTemplate=(uint8_t)0x38; } else if(model==DOG_LCD_M163) { //8-bit,2-line instructionSetTemplate=(uint8_t)0x38; } //init data-size and lines, then change to instructionset 1 setInstructionSet(1); //bias 1/4 writeCommand(0x1D,30); setContrast(this->contrast); //Standard setting is : display on, cursor on, no blink displayMode=0x04; cursorMode=0x02; blinkMode=0x00; writeDisplayMode(); entryMode=0x04; clear(); leftToRight(); } void DogLcd::setContrast(int contrast) { if(contrast<0 || contrast>0x3F) return; if(this->vcc==DOG_LCD_VCC_5V) { /* For 5v operation the booster must be off, which is on the same command as the (2-bit) high-nibble of contrast */ writeCommand((0x50 | ((contrast>>4)&0x03)),30); /* Set amplification ratio for the follower control */ writeCommand(0x69,30); } else { /* For 3.3v operation the booster must be on, which is on the same command as the (2-bit) high-nibble of contrast */ writeCommand((0x54 | ((contrast>>4)&0x03)),30); /* Set amplification ratio for the follower control this has to be higher it seems at 3.3V operation */ writeCommand(0x6B,30); } //set low-nibble of the contrast writeCommand((0x70 | (contrast & 0x0F)),30); } void DogLcd::clear() { writeCommand(0x01,1080); } void DogLcd::home() { writeCommand(0x02,1080); } void DogLcd::setCursor(int col, int row) { if(col>=memSize || row>=rows) { //not a valid cursor position return; } int address=(startAddress[row]+col) & 0x7F; writeCommand(0x80|address,30); } void DogLcd::noDisplay() { displayMode=0x00; writeDisplayMode(); } void DogLcd::display() { displayMode=0x04; writeDisplayMode(); } void DogLcd::noCursor() { cursorMode=0x00; writeDisplayMode(); } void DogLcd::cursor() { cursorMode=0x02; writeDisplayMode(); } void DogLcd::noBlink() { blinkMode=0x00; writeDisplayMode(); } void DogLcd::blink() { blinkMode=0x01; writeDisplayMode(); } void DogLcd::scrollDisplayLeft(void) { setInstructionSet(0); writeCommand(0x18,30); } void DogLcd::scrollDisplayRight(void) { setInstructionSet(0); writeCommand(0x1C,30); } void DogLcd::leftToRight(void) { entryMode|=0x02; writeCommand(entryMode,30); } void DogLcd::rightToLeft(void) { entryMode&=~0x02; writeCommand(entryMode,30); } void DogLcd::autoscroll(void) { entryMode|=0x01; writeCommand(entryMode,30); } void DogLcd::noAutoscroll(void) { entryMode&=~0x01; writeCommand(entryMode,30); } void DogLcd::createChar(int charPos, uint8_t charMap[]) { int baseAddress; if(charPos<0 || charPos>7) return; baseAddress=charPos*8; //changing CGRAM address belongs to different instruction set setInstructionSet(0); for (int i=0; i<8; i++) { writeCommand((0x40|(baseAddress+i)),30); writeChar(charMap[i]); } setInstructionSet(0); writeDisplayMode(); } void DogLcd::writeDisplayMode() { writeCommand((0x08 | displayMode | cursorMode | blinkMode),30); } void DogLcd::setBacklight(int value, bool usePWM) { if(backLight!=-1 && value>=0) { if(!usePWM) { if(value==LOW) { digitalWrite(backLight,LOW); } else { digitalWrite(backLight,HIGH); } } else { if(value>255) value=255; analogWrite(backLight,value); } } } void DogLcd::setInstructionSet(int is) { if(is<0 || is>3) return; int cmd=instructionSetTemplate | is; writeCommand(cmd,30); } void DogLcd::writeChar(int value) { digitalWrite(lcdRS,HIGH); spiTransfer(value,30); } void DogLcd::writeCommand(int value,int executionTime) { digitalWrite(lcdRS,LOW); spiTransfer(value,executionTime); } void DogLcd::spiTransfer(int value, int executionTime) { digitalWrite(lcdCLK,HIGH); digitalWrite(lcdCSB,LOW); for(int i=7;i>=0;i--) { if(bitRead(value,i)) { digitalWrite(lcdSI,HIGH); } else { digitalWrite(lcdSI,LOW); } digitalWrite(lcdCLK,LOW); digitalWrite(lcdCLK,HIGH); } digitalWrite(lcdCSB,HIGH); delayMicroseconds(executionTime); }