Added support for 7.5" v3 ePaper displays
This commit is contained in:
		| @@ -36,7 +36,7 @@ Inkycal v2.0.0 BETA is a refactoring of the previous release. It aims to fix cer | ||||
| * [x] Add info-section option in web-ui | ||||
| * [ ] Allow easy adding/removing of third-party modules | ||||
| * [ ] Finish developer documentation for third-party modules | ||||
| * [ ] Add official support for 7.5" v3 ePaper model (black-white + black-white-red/yellow) | ||||
| * [x] Add official support for 7.5" v3 ePaper model (black-white + black-white-red/yellow) | ||||
|  | ||||
| ## How to test BETA | ||||
| Please note that while inkycal is in BETA, a lot of things will change in a short time. This means that problems are fixed on-the-go. If you encounter a problem, please mention it on Discord. | ||||
|   | ||||
| @@ -24,6 +24,8 @@ class Layout: | ||||
|     if (model != None) and (width == None) and (height == None): | ||||
|       display_dimensions = { | ||||
|         '9_in_7': (1200, 825), | ||||
|         'epd_7_in_5_v3_colour': (880, 528), | ||||
|         'epd_7_in_5_v3': (880, 528), | ||||
|         'epd_7_in_5_v2_colour': (800, 480), | ||||
|         'epd_7_in_5_v2': (800, 480), | ||||
|         'epd_7_in_5_colour': (640, 384), | ||||
|   | ||||
| @@ -23,6 +23,7 @@ class Settings: | ||||
|   _supported_update_interval = [10, 15, 20, 30, 60] | ||||
|   _supported_display_orientation = ['normal', 'upside_down'] | ||||
|   _supported_models = [ | ||||
|   'epd_7_in_5_v3_colour', 'epd_7_in_5_v3', | ||||
|   'epd_7_in_5_v2_colour', 'epd_7_in_5_v2', | ||||
|   'epd_7_in_5_colour', 'epd_7_in_5', | ||||
|   'epd_5_in_83_colour','epd_5_in_83', | ||||
|   | ||||
							
								
								
									
										201
									
								
								inkycal/display/drivers/epd_7_in_5_v3.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										201
									
								
								inkycal/display/drivers/epd_7_in_5_v3.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,201 @@ | ||||
| # ***************************************************************************** | ||||
| # * | File        :	  epd7in5.py | ||||
| # * | Author      :   Waveshare team | ||||
| # * | Function    :   Electronic paper driver | ||||
| # * | Info        : | ||||
| # *---------------- | ||||
| # * | This version:   V4.0 | ||||
| # * | Date        :   2019-06-20 | ||||
| # # | Info        :   python demo | ||||
| # ----------------------------------------------------------------------------- | ||||
| # Permission is hereby granted, free of charge, to any person obtaining a copy | ||||
| # of this software and associated documnetation files (the "Software"), to deal | ||||
| # in the Software without restriction, including without limitation the rights | ||||
| # to use, copy, modify, merge, publish, distribute, sublicense, and/or sell | ||||
| # copies of the Software, and to permit persons to  whom the Software is | ||||
| # furished to do so, subject to the following conditions: | ||||
| # | ||||
| # The above copyright notice and this permission notice shall be included in | ||||
| # all copies or substantial portions of the Software. | ||||
| # | ||||
| # THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | ||||
| # IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | ||||
| # FITNESS OR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE | ||||
| # AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER | ||||
| # LIABILITY WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, | ||||
| # OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN | ||||
| # THE SOFTWARE. | ||||
| # | ||||
|  | ||||
|  | ||||
| import logging | ||||
| from . import epdconfig | ||||
|  | ||||
| # Display resolution | ||||
| EPD_WIDTH       = 880 | ||||
| EPD_HEIGHT      = 528 | ||||
|  | ||||
| class EPD: | ||||
|     def __init__(self): | ||||
|         self.reset_pin = epdconfig.RST_PIN | ||||
|         self.dc_pin = epdconfig.DC_PIN | ||||
|         self.busy_pin = epdconfig.BUSY_PIN | ||||
|         self.cs_pin = epdconfig.CS_PIN | ||||
|         self.width = EPD_WIDTH | ||||
|         self.height = EPD_HEIGHT | ||||
|      | ||||
|     # Hardware reset | ||||
|     def reset(self): | ||||
|         epdconfig.digital_write(self.reset_pin, 1) | ||||
|         epdconfig.delay_ms(200)  | ||||
|         epdconfig.digital_write(self.reset_pin, 0) | ||||
|         epdconfig.delay_ms(2) | ||||
|         epdconfig.digital_write(self.reset_pin, 1) | ||||
|         epdconfig.delay_ms(200)    | ||||
|  | ||||
|     def send_command(self, command): | ||||
|         epdconfig.digital_write(self.dc_pin, 0) | ||||
|         epdconfig.digital_write(self.cs_pin, 0) | ||||
|         epdconfig.spi_writebyte([command]) | ||||
|         epdconfig.digital_write(self.cs_pin, 1) | ||||
|  | ||||
|     def send_data(self, data): | ||||
|         epdconfig.digital_write(self.dc_pin, 1) | ||||
|         epdconfig.digital_write(self.cs_pin, 0) | ||||
|         epdconfig.spi_writebyte([data]) | ||||
|         epdconfig.digital_write(self.cs_pin, 1) | ||||
|          | ||||
|     def ReadBusy(self): | ||||
|         logging.debug("e-Paper busy") | ||||
|         busy = epdconfig.digital_read(self.busy_pin) | ||||
|         while(busy == 1): | ||||
|             busy = epdconfig.digital_read(self.busy_pin) | ||||
|         epdconfig.delay_ms(200) | ||||
|          | ||||
|     def init(self): | ||||
|         if (epdconfig.module_init() != 0): | ||||
|             return -1 | ||||
|         # EPD hardware init start | ||||
|         self.reset() | ||||
|          | ||||
|         self.ReadBusy(); | ||||
|         self.send_command(0x12);  #SWRESET | ||||
|         self.ReadBusy(); | ||||
|  | ||||
|         self.send_command(0x46);  # Auto Write Red RAM | ||||
|         self.send_data(0xf7); | ||||
|         self.ReadBusy(); | ||||
|         self.send_command(0x47);  # Auto Write  B/W RAM | ||||
|         self.send_data(0xf7); | ||||
|         self.ReadBusy(); | ||||
|  | ||||
|         self.send_command(0x0C);  # Soft start setting | ||||
|         self.send_data(0xAE); | ||||
|         self.send_data(0xC7); | ||||
|         self.send_data(0xC3); | ||||
|         self.send_data(0xC0); | ||||
|         self.send_data(0x40);  | ||||
|  | ||||
|  | ||||
|         self.send_command(0x01);  # Set MUX as 527 | ||||
|         self.send_data(0xAF); | ||||
|         self.send_data(0x02); | ||||
|         self.send_data(0x01);#0x01 | ||||
|  | ||||
|         self.send_command(0x11);  # Data entry mode       | ||||
|         self.send_data(0x01); | ||||
|  | ||||
|         self.send_command(0x44);  | ||||
|         self.send_data(0x00); # RAM x address start at 0 | ||||
|         self.send_data(0x00);  | ||||
|         self.send_data(0x6F);  | ||||
|         self.send_data(0x03);  | ||||
|         self.send_command(0x45);  | ||||
|         self.send_data(0xAF);  | ||||
|         self.send_data(0x02); | ||||
|         self.send_data(0x00);  | ||||
|         self.send_data(0x00); | ||||
|  | ||||
|         self.send_command(0x3C); # VBD | ||||
|         self.send_data(0x05); # LUT1, for white | ||||
|  | ||||
|         self.send_command(0x18); | ||||
|         self.send_data(0X80); | ||||
|  | ||||
|  | ||||
|         self.send_command(0x22); | ||||
|         self.send_data(0XB1); #Load Temperature and waveform setting. | ||||
|         self.send_command(0x20); | ||||
|         self.ReadBusy(); | ||||
|  | ||||
|         self.send_command(0x4E); # set RAM x address count to 0; | ||||
|         self.send_data(0x00); | ||||
|         self.send_data(0x00); | ||||
|         self.send_command(0x4F);  | ||||
|         self.send_data(0x00); | ||||
|         self.send_data(0x00); | ||||
|         # EPD hardware init end | ||||
|         return 0 | ||||
|  | ||||
|     def getbuffer(self, image): | ||||
|         # logging.debug("bufsiz = ",int(self.width/8) * self.height) | ||||
|         buf = [0xFF] * (int(self.width/8) * self.height) | ||||
|         image_monocolor = image.convert('1') | ||||
|         imwidth, imheight = image_monocolor.size | ||||
|         pixels = image_monocolor.load() | ||||
|         # logging.debug("imwidth = %d, imheight = %d",imwidth,imheight) | ||||
|         if(imwidth == self.width and imheight == self.height): | ||||
|             logging.debug("Vertical") | ||||
|             for y in range(imheight): | ||||
|                 for x in range(imwidth): | ||||
|                     # Set the bits for the column of pixels at the current position. | ||||
|                     if pixels[x, y] == 0: | ||||
|                         buf[int((x + y * self.width) / 8)] &= ~(0x80 >> (x % 8)) | ||||
|         elif(imwidth == self.height and imheight == self.width): | ||||
|             logging.debug("Horizontal") | ||||
|             for y in range(imheight): | ||||
|                 for x in range(imwidth): | ||||
|                     newx = y | ||||
|                     newy = self.height - x - 1 | ||||
|                     if pixels[x, y] == 0: | ||||
|                         buf[int((newx + newy*self.width) / 8)] &= ~(0x80 >> (y % 8)) | ||||
|         return buf | ||||
|          | ||||
|     def display(self, image): | ||||
|         self.send_command(0x4F);  | ||||
|         self.send_data(0x00); | ||||
|         self.send_data(0x00); | ||||
|         self.send_command(0x24); | ||||
|         for i in range(0, int(self.width * self.height / 8)): | ||||
|             self.send_data(image[i]); | ||||
|                  | ||||
|         self.send_command(0x22); | ||||
|         self.send_data(0xF7);#Load LUT from MCU(0x32) | ||||
|         self.send_command(0x20); | ||||
|         epdconfig.delay_ms(10); | ||||
|         self.ReadBusy(); | ||||
|          | ||||
|     def Clear(self): | ||||
|         self.send_command(0x4F);  | ||||
|         self.send_data(0x00); | ||||
|         self.send_data(0x00); | ||||
|         self.send_command(0x24) | ||||
|         for i in range(0, int(self.width * self.height / 8)): | ||||
|             self.send_data(0xff) | ||||
|              | ||||
|         self.send_command(0x26) | ||||
|         for i in range(0, int(self.width * self.height / 8)): | ||||
|             self.send_data(0xff) | ||||
|                  | ||||
|         self.send_command(0x22); | ||||
|         self.send_data(0xF7);#Load LUT from MCU(0x32) | ||||
|         self.send_command(0x20); | ||||
|         epdconfig.delay_ms(10); | ||||
|         self.ReadBusy(); | ||||
|  | ||||
|     def sleep(self): | ||||
|         self.send_command(0x10); | ||||
|         self.send_data(0x01); | ||||
|          | ||||
|         epdconfig.module_exit() | ||||
| ### END OF FILE ### | ||||
							
								
								
									
										204
									
								
								inkycal/display/drivers/epd_7_in_5_v3_colour.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										204
									
								
								inkycal/display/drivers/epd_7_in_5_v3_colour.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,204 @@ | ||||
| # ***************************************************************************** | ||||
| # * | File        :	  epd7in5bc.py | ||||
| # * | Author      :   Waveshare team | ||||
| # * | Function    :   Electronic paper driver | ||||
| # * | Info        : | ||||
| # *---------------- | ||||
| # * | This version:   V4.0 | ||||
| # * | Date        :   2019-06-20 | ||||
| # # | Info        :   python demo | ||||
| # ----------------------------------------------------------------------------- | ||||
| # Permission is hereby granted, free of charge, to any person obtaining a copy | ||||
| # of this software and associated documnetation files (the "Software"), to deal | ||||
| # in the Software without restriction, including without limitation the rights | ||||
| # to use, copy, modify, merge, publish, distribute, sublicense, and/or sell | ||||
| # copies of the Software, and to permit persons to  whom the Software is | ||||
| # furished to do so, subject to the following conditions: | ||||
| # | ||||
| # The above copyright notice and this permission notice shall be included in | ||||
| # all copies or substantial portions of the Software. | ||||
| # | ||||
| # THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | ||||
| # IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | ||||
| # FITNESS OR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE | ||||
| # AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER | ||||
| # LIABILITY WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, | ||||
| # OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN | ||||
| # THE SOFTWARE. | ||||
| # | ||||
|  | ||||
|  | ||||
| import logging | ||||
| from inkycal.display.drivers import epdconfig | ||||
|  | ||||
| # Display resolution | ||||
| EPD_WIDTH       = 880 | ||||
| EPD_HEIGHT      = 528 | ||||
|  | ||||
| class EPD: | ||||
|     def __init__(self): | ||||
|         self.reset_pin = epdconfig.RST_PIN | ||||
|         self.dc_pin = epdconfig.DC_PIN | ||||
|         self.busy_pin = epdconfig.BUSY_PIN | ||||
|         self.cs_pin = epdconfig.CS_PIN | ||||
|         self.width = EPD_WIDTH | ||||
|         self.height = EPD_HEIGHT | ||||
|  | ||||
|     # Hardware reset | ||||
|     def reset(self): | ||||
|         epdconfig.digital_write(self.reset_pin, 1) | ||||
|         epdconfig.delay_ms(200)  | ||||
|         epdconfig.digital_write(self.reset_pin, 0) | ||||
|         epdconfig.delay_ms(4) | ||||
|         epdconfig.digital_write(self.reset_pin, 1) | ||||
|         epdconfig.delay_ms(200)    | ||||
|  | ||||
|     def send_command(self, command): | ||||
|         epdconfig.digital_write(self.dc_pin, 0) | ||||
|         epdconfig.digital_write(self.cs_pin, 0) | ||||
|         epdconfig.spi_writebyte([command]) | ||||
|         epdconfig.digital_write(self.cs_pin, 1) | ||||
|  | ||||
|     def send_data(self, data): | ||||
|         epdconfig.digital_write(self.dc_pin, 1) | ||||
|         epdconfig.digital_write(self.cs_pin, 0) | ||||
|         epdconfig.spi_writebyte([data]) | ||||
|         epdconfig.digital_write(self.cs_pin, 1) | ||||
|          | ||||
|     def ReadBusy(self): | ||||
|         logging.debug("e-Paper busy") | ||||
|         busy = epdconfig.digital_read(self.busy_pin) | ||||
|         while(busy == 1): | ||||
|             busy = epdconfig.digital_read(self.busy_pin) | ||||
|         epdconfig.delay_ms(200) | ||||
|              | ||||
|     def init(self): | ||||
|         if (epdconfig.module_init() != 0): | ||||
|             return -1 | ||||
|              | ||||
|         self.reset() | ||||
|          | ||||
|         self.send_command(0x12); 		  #SWRESET | ||||
|         self.ReadBusy();        #waiting for the electronic paper IC to release the idle signal | ||||
|  | ||||
|         self.send_command(0x46);  # Auto Write RAM | ||||
|         self.send_data(0xF7); | ||||
|         self.ReadBusy();        #waiting for the electronic paper IC to release the idle signal | ||||
|  | ||||
|         self.send_command(0x47);  # Auto Write RAM | ||||
|         self.send_data(0xF7); | ||||
|         self.ReadBusy();        #waiting for the electronic paper IC to release the idle signal | ||||
|  | ||||
|         self.send_command(0x0C);  # Soft start setting | ||||
|         self.send_data(0xAE); | ||||
|         self.send_data(0xC7); | ||||
|         self.send_data(0xC3); | ||||
|         self.send_data(0xC0); | ||||
|         self.send_data(0x40);    | ||||
|  | ||||
|         self.send_command(0x01);  # Set MUX as 527 | ||||
|         self.send_data(0xAF); | ||||
|         self.send_data(0x02); | ||||
|         self.send_data(0x01); | ||||
|  | ||||
|         self.send_command(0x11);  # Data entry mode | ||||
|         self.send_data(0x01); | ||||
|  | ||||
|         self.send_command(0x44); | ||||
|         self.send_data(0x00); # RAM x address start at 0 | ||||
|         self.send_data(0x00); | ||||
|         self.send_data(0x6F); # RAM x address end at 36Fh -> 879 | ||||
|         self.send_data(0x03); | ||||
|         self.send_command(0x45); | ||||
|         self.send_data(0xAF); # RAM y address start at 20Fh; | ||||
|         self.send_data(0x02); | ||||
|         self.send_data(0x00); # RAM y address end at 00h; | ||||
|         self.send_data(0x00); | ||||
|  | ||||
|         self.send_command(0x3C); # VBD | ||||
|         self.send_data(0x01); # LUT1, for white | ||||
|  | ||||
|         self.send_command(0x18); | ||||
|         self.send_data(0X80); | ||||
|         self.send_command(0x22); | ||||
|         self.send_data(0XB1);	#Load Temperature and waveform setting. | ||||
|         self.send_command(0x20); | ||||
|         self.ReadBusy();        #waiting for the electronic paper IC to release the idle signal | ||||
|  | ||||
|         self.send_command(0x4E);  | ||||
|         self.send_data(0x00); | ||||
|         self.send_data(0x00); | ||||
|         self.send_command(0x4F);  | ||||
|         self.send_data(0xAF); | ||||
|         self.send_data(0x02); | ||||
|          | ||||
|         return 0 | ||||
|  | ||||
|     def getbuffer(self, image): | ||||
|         # logging.debug("bufsiz = ",int(self.width/8) * self.height) | ||||
|         buf = [0xFF] * (int(self.width/8) * self.height) | ||||
|         image_monocolor = image.convert('1') | ||||
|         imwidth, imheight = image_monocolor.size | ||||
|         pixels = image_monocolor.load() | ||||
|         logging.debug('imwidth = %d  imheight =  %d ',imwidth, imheight) | ||||
|         if(imwidth == self.width and imheight == self.height): | ||||
|             logging.debug("Horizontal") | ||||
|             for y in range(imheight): | ||||
|                 for x in range(imwidth): | ||||
|                     # Set the bits for the column of pixels at the current position. | ||||
|                     if pixels[x, y] == 0: | ||||
|                         buf[int((x + y * self.width) / 8)] &= ~(0x80 >> (x % 8)) | ||||
|         elif(imwidth == self.height and imheight == self.width): | ||||
|             logging.debug("Vertical") | ||||
|             for y in range(imheight): | ||||
|                 for x in range(imwidth): | ||||
|                     newx = y | ||||
|                     newy = self.height - x - 1 | ||||
|                     if pixels[x, y] == 0: | ||||
|                         buf[int((newx + newy*self.width) / 8)] &= ~(0x80 >> (y % 8)) | ||||
|         return buf | ||||
|  | ||||
|     def display(self, imageblack, imagered): | ||||
|         self.send_command(0x4F);  | ||||
|         self.send_data(0xAf); | ||||
|          | ||||
|         self.send_command(0x24) | ||||
|         for i in range(0, int(self.width * self.height / 8)): | ||||
|             self.send_data(imageblack[i]); | ||||
|          | ||||
|          | ||||
|         self.send_command(0x26) | ||||
|         for i in range(0, int(self.width * self.height / 8)): | ||||
|             self.send_data(~imagered[i]); | ||||
|          | ||||
|         self.send_command(0x22); | ||||
|         self.send_data(0xC7);    #Load LUT from MCU(0x32) | ||||
|         self.send_command(0x20); | ||||
|         epdconfig.delay_ms(200);      #!!!The delay here is necessary, 200uS at least!!!      | ||||
|         self.ReadBusy(); | ||||
|          | ||||
|     def Clear(self): | ||||
|         self.send_command(0x4F);  | ||||
|         self.send_data(0xAf); | ||||
|          | ||||
|         self.send_command(0x24) | ||||
|         for i in range(0, int(self.width * self.height / 8)): | ||||
|             self.send_data(0xff); | ||||
|          | ||||
|          | ||||
|         self.send_command(0x26) | ||||
|         for i in range(0, int(self.width * self.height / 8)): | ||||
|             self.send_data(0x00); | ||||
|          | ||||
|         self.send_command(0x22); | ||||
|         self.send_data(0xC7);    #Load LUT from MCU(0x32) | ||||
|         self.send_command(0x20); | ||||
|         epdconfig.delay_ms(200);      #!!!The delay here is necessary, 200uS at least!!!      | ||||
|         self.ReadBusy(); | ||||
|  | ||||
|     def sleep(self): | ||||
|         self.send_command(0x10);  	#deep sleep | ||||
|         self.send_data(0x01); | ||||
|          | ||||
|         epdconfig.module_exit() | ||||
| ### END OF FILE ### | ||||
| @@ -84,6 +84,15 @@ | ||||
|               <input id="9_in_7" type="radio" name="dp" checked> | ||||
|               <label for="9_in_7">9.7" ePaper</label> | ||||
|             </div> | ||||
|           <div class="ts checkboxes"> | ||||
|             <div class="ts radio checkbox"> | ||||
|               <input id="epd_7_in_5_v3_colour" type="radio" name="dp" checked> | ||||
|               <label for="epd_7_in_5_v3_colour">7.5" v3 (880x528px) colour (latest)</label> | ||||
|             </div> | ||||
|             <div class="ts radio checkbox"> | ||||
|               <input id="epd_7_in_5_v3" type="radio" name="dp"> | ||||
|               <label for="epd_7_in_5_v3">7.5" v3 (880x528px) black-white (latest)</label> | ||||
|             </div> | ||||
|           <div class="ts checkboxes"> | ||||
|             <div class="ts radio checkbox"> | ||||
|               <input id="epd_7_in_5_v2_colour" type="radio" name="dp" checked> | ||||
| @@ -543,12 +552,18 @@ | ||||
|         calibration_hours = $("#calibration_hours").attr("placeholder"); | ||||
|       } | ||||
|  | ||||
|       var model = "epd_7_in_5_v2_colour"; | ||||
|       var model = "epd_7_in_5_v2"; | ||||
|       if ($('#9_in_7').is(':checked')) { | ||||
|         model = "9_in_7"; | ||||
|       } | ||||
|       if ($('#epd_7_in_5_v2').is(':checked')) { | ||||
|         model = "epd_7_in_5_v2"; | ||||
|       if ($('#epd_7_in_5_v3_colour').is(':checked')) { | ||||
|         model = "epd_7_in_5_v3_colour"; | ||||
|       } | ||||
|       if ($('#epd_7_in_5_v3').is(':checked')) { | ||||
|         model = "epd_7_in_5_v3"; | ||||
|       } | ||||
|       if ($('#epd_7_in_5_v2_colour').is(':checked')) { | ||||
|         model = "epd_7_in_5_v2_colour"; | ||||
|       } | ||||
|       if ($('#epd_7_in_5_colour').is(':checked')) { | ||||
|         model = "epd_7_in_5_colour"; | ||||
| @@ -819,4 +834,4 @@ | ||||
|   </script> | ||||
| </body> | ||||
|  | ||||
| </html> | ||||
| </html> | ||||
		Reference in New Issue
	
	Block a user