# -*- coding: utf-8 -*-
# Original code found at:
# https://gist.github.com/DenisFromHR/cc863375a6e19dce359d

"""
Compiled, mashed and generally mutilated 2014-2015 by Denis Pleic
Made available under GNU GENERAL PUBLIC LICENSE

# Modified Python I2C library for Raspberry Pi
# as found on http://www.recantha.co.uk/blog/?p=4849
# Joined existing 'i2c_lib.py' and 'lcddriver.py' into a single library
# added bits and pieces from various sources
# By DenisFromHR (Denis Pleic)
# 2015-02-10, ver 0.1

"""

# i2c bus (0 -- original Pi, 1 -- Rev 2 Pi)
I2CBUS 1

# LCD Address
ADDRESS 0x3f

import smbus
from time import sleep

class i2c_device:
   def __init__(selfaddrport=I2CBUS):
      self.addr addr
      self.bus smbus.SMBus(port)

# Write a single command
   def write_cmd(selfcmd):
      self.bus.write_byte(self.addrcmd)
      sleep(0.0001)

# Write a command and argument
   def write_cmd_arg(selfcmddata):
      self.bus.write_byte_data(self.addrcmddata)
      sleep(0.0001)

# Write a block of data
   def write_block_data(selfcmddata):
      self.bus.write_block_data(self.addrcmddata)
      sleep(0.0001)

# Read a single byte
   def read(self):
      return self.bus.read_byte(self.addr)

# Read
   def read_data(selfcmd):
      return self.bus.read_byte_data(self.addrcmd)

# Read a block of data
   def read_block_data(selfcmd):
      return self.bus.read_block_data(self.addrcmd)


# commands
LCD_CLEARDISPLAY 0x01
LCD_RETURNHOME 0x02
LCD_ENTRYMODESET 0x04
LCD_DISPLAYCONTROL 0x08
LCD_CURSORSHIFT 0x10
LCD_FUNCTIONSET 0x20
LCD_SETCGRAMADDR 0x40
LCD_SETDDRAMADDR 0x80

# flags for display entry mode
LCD_ENTRYRIGHT 0x00
LCD_ENTRYLEFT 0x02
LCD_ENTRYSHIFTINCREMENT 0x01
LCD_ENTRYSHIFTDECREMENT 0x00

# flags for display on/off control
LCD_DISPLAYON 0x04
LCD_DISPLAYOFF 0x00
LCD_CURSORON 0x02
LCD_CURSOROFF 0x00
LCD_BLINKON 0x01
LCD_BLINKOFF 0x00

# flags for display/cursor shift
LCD_DISPLAYMOVE 0x08
LCD_CURSORMOVE 0x00
LCD_MOVERIGHT 0x04
LCD_MOVELEFT 0x00

# flags for function set
LCD_8BITMODE 0x10
LCD_4BITMODE 0x00
LCD_2LINE 0x08
LCD_1LINE 0x00
LCD_5x10DOTS 0x04
LCD_5x8DOTS 0x00

# flags for backlight control
LCD_BACKLIGHT 0x08
LCD_NOBACKLIGHT 0x00

En 0b00000100 # Enable bit
Rw 0b00000010 # Read/Write bit
Rs 0b00000001 # Register select bit

class lcd:
   #initializes objects and lcd
   def __init__(self):
      self.lcd_device i2c_device(ADDRESS)

      self.lcd_write(0x03)
      self.lcd_write(0x03)
      self.lcd_write(0x03)
      self.lcd_write(0x02)

      self.lcd_write(LCD_FUNCTIONSET LCD_2LINE LCD_5x8DOTS LCD_4BITMODE)
      self.lcd_write(LCD_DISPLAYCONTROL LCD_DISPLAYON)
      self.lcd_write(LCD_CLEARDISPLAY)
      self.lcd_write(LCD_ENTRYMODESET LCD_ENTRYLEFT)
      sleep(0.2)


   # clocks EN to latch command
   def lcd_strobe(selfdata):
      self.lcd_device.write_cmd(data En LCD_BACKLIGHT)
      sleep(.0005)
      self.lcd_device.write_cmd(((data & ~En) | LCD_BACKLIGHT))
      sleep(.0001)

   def lcd_write_four_bits(selfdata):
      self.lcd_device.write_cmd(data LCD_BACKLIGHT)
      self.lcd_strobe(data)

   # write a command to lcd
   def lcd_write(selfcmdmode=0):
      self.lcd_write_four_bits(mode | (cmd 0xF0))
      self.lcd_write_four_bits(mode | ((cmd << 4) & 0xF0))

   # write a character to lcd (or character rom) 0x09: backlight | RS=DR<
   # works!
   def lcd_write_char(selfcharvaluemode=1):
      self.lcd_write_four_bits(mode | (charvalue 0xF0))
      self.lcd_write_four_bits(mode | ((charvalue << 4) & 0xF0))
  
   # put string function with optional char positioning
   def lcd_display_string(selfstringline=1pos=0):
    if line == 1:
      pos_new pos
    elif line == 2:
      pos_new 0x40 pos
    elif line == 3:
      pos_new 0x14 pos
    elif line == 4:
      pos_new 0x54 pos

    self.lcd_write(0x80 pos_new)

    for char in string:
      self.lcd_write(ord(char), Rs)

   # clear lcd and set to home
   def lcd_clear(self):
      self.lcd_write(LCD_CLEARDISPLAY)
      self.lcd_write(LCD_RETURNHOME)

   # define backlight on/off (lcd.backlight(1); off= lcd.backlight(0)
   def backlight(selfstate): # for state, 1 = on, 0 = off
      if state == 1:
         self.lcd_device.write_cmd(LCD_BACKLIGHT)
      elif state == 0:
         self.lcd_device.write_cmd(LCD_NOBACKLIGHT)

   # add custom characters (0 - 7)
   def lcd_load_custom_chars(selffontdata):
      self.lcd_write(0x40);
      for char in fontdata:
         for line in char:
            self.lcd_write_char(line)