====== Elecrow CrowPanel 5.79" E-Paper display ====== E-Paper display with 792*272 resolution, using ESP32-S3 as the main control, powerful performance. {{:components:elecrow_crowpanel_5.79_e-paper_display.png?400|}} Specification |Panel size|168mm * 57mm| |Screen size|5.79 inch| |Display Color|Black and white| |Resolution|792(L) * 272(H) Pixel| |Pixel pitch|0.1755*0.1755| |Viewing Angle|Full Viewing Angle| |MCU|ESP32-S3-WROOM-1-N8R8,up to 240 MHz| |Flash|8 MB| |PSRAM|8 MB| |Material|Active Matrix Electroph oretic Display (AM EPD)| |Driver Chip|SSD1683 * 2| |Communication interface|3-/4-wire SPI, default 4-wire SPI| |Interface|UART0x1, BATx1, GPIOx1, TF Card Slot x1| |Button|Dial Switchx1, Menu Buttonx1, Back Buttonx1, REST Buttonx1, BOOT Buttonx1| |Development Environment|Arduino IDE、ESP IDF、MicroPython| |Refresh Mode|Partial refresh (saves more power)| |Display Voltage|2.2~3.7V| |Operation Temperature|-0~50℃| |Storage Temperature|-25~70℃| |Active Area|47.74(H)*139.00(L)(H*L)| ====== Code ====== ===== Library ===== # This library CrowPanel E-paper display, based on SSD1683 chips # CrowPanel 5.79" # CrowPanel 4.20" (not tested) # # Library is based on micropython frameBuffer # # V0.1.0 Dec 2025 Initial version # V0.1.1 Dec 2025 Fixed initialization of screen # # Released under the MIT License (MIT). # Copyright (c) 2025 Ignas Bukys from micropython import const from time import sleep_ms import framebuf from ustruct import pack from io import BytesIO from machine import SPI, Pin __version__ = (0, 1, 1) # Display colour codes COLOR_WHITE = const(1) COLOR_BLACK = const(0) #generic class for chip class SSD1683(framebuf.FrameBuffer): '''Low-level controls for E-Paper chip''' # Constants for SSD1608 driver IC SET_DRIVER_CONTROL = const(0x01) SET_GATE_VOLTAGE = const(0x03) SET_SOURCE_VOLTAGE = const(0x04) SET_DISPLAY_CONTROL = const(0x07) SET_NON_OVERLAP = const(0x0B) SET_BOOSTER_SOFT_START = const(0x0C) SET_GATE_SCAN_START = const(0x0F) SET_DEEP_SLEEP = const(0x10) SET_DATA_MODE = const(0x11) SET_DATA_MODE_SLAVE = const(0x91) SET_SW_RESET = const(0x12) SET_TEMP_WRITE = const(0x1A) SET_TEMP_READ = const(0x1B) SET_TEMP_CONTROL = const(0x18) SET_TEMP_LOAD = const(0x1A) SET_MASTER_ACTIVATE = const(0x20) SET_DISP_CTRL1 = const(0x21) SET_DISP_CTRL2 = const(0x22) SET_WRITE_RAM = const(0x24) SET_WRITE_ALTRAM = const(0x26) SET_READ_RAM = const(0x25) SET_VCOM_SENSE = const(0x2B) SET_VCOM_DURATION = const(0x2C) SET_WRITE_VCOM = const(0x2C) SET_READ_OTP = const(0x2D) SET_WRITE_LUT = const(0x32) SET_WRITE_DUMMY = const(0x3A) SET_WRITE_GATELINE = const(0x3B) SET_WRITE_BORDER = const(0x3C) SET_RAMXPOS = const(0x44) SET_RAMYPOS = const(0x45) SET_RAMXCOUNT = const(0x4E) SET_RAMYCOUNT = const(0x4F) SET_WRITE_RAM_SLAVE = const(0xA4) SET_WRITE_ALTRAM_SLAVE = const(0xA6) SET_RAMXPOS_SLAVE = const(0xC4) SET_RAMYPOS_SLAVE = const(0xC5) SET_RAMXCOUNT_SLAVE = const(0xCE) SET_RAMYCOUNT_SLAVE = const(0xCF) SET_NOP = const(0xFF) # Pins for communication LED_PIN = 41 RESET_PIN = 47 BUSY_PIN = 48 DC_PIN = 46 MOSI_PIN = 11 SCK_PIN = 12 CS_PIN = 45 SCREEN_POWER_PIN = 7 # Rotation ROTATION_0 = const(0) ROTATION_90 = const(1) ROTATION_180 = const(2) ROTATION_270 = const(3) def __init__(self, w, h, rotation=ROTATION_0): self._init_spi() self._init_buffer(w, h, rotation) self.FastMode1Init() self.HW_RESET() def _init_spi(self): #Set pin 7 to high level to activate the screen power Pin(self.SCREEN_POWER_PIN, Pin.OUT, value=1) self.cs = Pin(self.CS_PIN, Pin.OUT) self.dc = Pin(self.DC_PIN, Pin.OUT) self.rst = Pin(self.RESET_PIN, Pin.OUT) self.busy = Pin(self.BUSY_PIN, Pin.IN) self.spi = SPI(1, baudrate=4_000_000, sck=Pin(self.SCK_PIN), mosi=Pin(self.MOSI_PIN), polarity=0, phase=0, firstbit=SPI.MSB) self.spi.init() self.cs.init(self.cs.OUT, value=1) self.dc.init(self.dc.OUT, value=1) self.rst.init(self.rst.OUT, value=1) self.busy.init(self.busy.IN, value=0) def _cmd(self, command, data=None): '''command and optional 1 byte of data''' self.cs(1) self.dc(0) self.cs(0) self.spi.write(bytearray([command])) self.cs(1) if data is not None: self._data(data) def _data(self, data): '''one byte of data''' self.cs(1) self.dc(1) self.cs(0) self.spi.write(bytearray([data])) self.cs(1) def _data_s(self, data): '''data in stream of bytes''' self.cs(1) self.dc(1) self.cs(0) self.spi.write(pack('B'*len(data), *data)) self.cs(1) def _init_buffer(self, w, h, rotation): self._rotation = rotation size = w * h // 8 self.buffer = bytearray(size) if self._rotation == self.ROTATION_0 or self._rotation == self.ROTATION_180: self.width = w self.height = h else: self.width = w self.height = h super().__init__(self.buffer, self.width, self.height, framebuf.MONO_HLSB) print('Buffer width:{}, height:{}, size:{}'.format(self.width, self.height, size)) def _wait_until_idle(self): while self.busy.value() == 1: sleep_ms(10) def HW_RESET(self): '''Perform Hardware reset''' sleep_ms(10) self.rst(0) sleep_ms(10) self.rst(1) sleep_ms(10) self._wait_until_idle() def EPD_Init(self): '''Intialize screen by doing Hard and Soft reset. Wait till busy pin low''' self.HW_RESET() self._wait_until_idle() self._cmd(self.SET_SW_RESET) self._wait_until_idle() def FastMode1Init(self): self.EPD_Init() self._cmd(self.SET_TEMP_CONTROL, 0x80) # Read built-in temperature sensor self._cmd(self.SET_DISP_CTRL2, 0xB1) # Load temperature value self._cmd(self.SET_MASTER_ACTIVATE) self._wait_until_idle() self._cmd(self.SET_TEMP_WRITE, 0x64) # Write to temperature register self._data(0x00) self._cmd(self.SET_DISP_CTRL2, 0x91) # Load temperature value self._cmd(self.SET_MASTER_ACTIVATE) self._wait_until_idle() self._cmd(self.SET_WRITE_BORDER, 0x1) # 0x3 | 0-ryškus 1-jokio 2-jokio self._wait_until_idle() def Display_Clear(self, count): '''Fill ram and altram of both chips with 0s and 1s''' self.SetRAMMP() self.SetRAMMA() self._cmd(self.SET_WRITE_RAM) self._data_s(b'\xFF' * count) self.SetRAMMA() self._cmd(self.SET_WRITE_ALTRAM) self._data_s(b'\x00' * count) self.SetRAMSP() self.SetRAMSA() self._cmd(self.SET_WRITE_RAM_SLAVE) self._data_s(b'\xFF' * count) self.SetRAMSA() self._cmd(self.SET_WRITE_ALTRAM_SLAVE) self._data_s(b'\x00' * count) def SetRAMMP(self): '''Data entry mode for ram primary''' self._cmd(self.SET_DATA_MODE, 0x02) # Data Entry mode setting; 1 –Y decrement, X increment self._cmd(self.SET_RAMXPOS) # Set Ram X- address Start / End position self._data(0x31) # XStart, POR = 00h self._data(0x00) self._cmd(self.SET_RAMYPOS) # Set Ram Y- address Start / End position self._data(0x00) self._data(0x00) self._data(0x0f) self._data(0x01) def SetRAMMA(self): '''Data entry mode for altram primary''' self._cmd(self.SET_RAMXCOUNT, 0x31) self._cmd(self.SET_RAMYCOUNT, 0x00) self._data(0x00) def SetRAMSP(self): '''Data entry mode for ram Slave''' self._cmd(self.SET_DATA_MODE_SLAVE, 0x03) self._cmd(self.SET_RAMXPOS_SLAVE) self._data(0x00) self._data(0x31) self._cmd(self.SET_RAMYPOS_SLAVE) self._data(0x00) # Set Ram Y- address Start / End position self._data(0x00) self._data(0x0f) # YEnd L self._data(0x01) def SetRAMSA(self): '''Data entry mode for altram Slave''' self._cmd(self.SET_RAMXCOUNT_SLAVE, 0x00) self._cmd(self.SET_RAMYCOUNT_SLAVE, 0x00) self._data(0x00) def Update(self): self._cmd(self.SET_DISP_CTRL2, 0xF7) self._cmd(self.SET_MASTER_ACTIVATE) self._wait_until_idle() def PartUpdate(self): self._cmd(self.SET_DISP_CTRL2, 0xDC) self._cmd(self.SET_MASTER_ACTIVATE) self._wait_until_idle() def FastUpdate(self): self._cmd(self.SET_DISP_CTRL2, 0xC7) self._cmd(self.SET_MASTER_ACTIVATE) self._wait_until_idle() def DeepSleep(self, mode=0x01): '''Send device to sleep and save power To wake up call EPD_Init() Parameters ---------- int: mode 0x00- Normal 0x01- Mode1 0x11- Mode2 (without RAM retention) ''' self._cmd(self.SET_DEEP_SLEEP, mode) sleep_ms(5) def LoadImage(self, PosX, PosY, ImgName, ImgWidth, ImgHeight): ''' Load image into frame buffer on predefined possition Your file must be Black and White. You can use https://javl.github.io/image2cpp/ Draw mode must be "Horizontal- 1 bit per pixel" framebuf.MONO_HLSB Parameters ---------- int : Possition X on buffer int : Possition Y on buffer str : Image location int : Image width int : Image height Raises ------ ValueError Sometime buffer construct method return it. I try to choose 2^X images of square dimensions ''' # Create a bytearray to store the image data img_data = bytearray(ImgWidth * ImgHeight // 8) with open(ImgName, 'rb') as f: f.readinto(img_data) # Create a FrameBuffer object from the image data img_buf = framebuf.FrameBuffer(img_data, ImgWidth, ImgHeight, framebuf.MONO_HLSB) self.blit(img_buf, PosX, PosY) class Screen_579(SSD1683): '''device specifics for CrowPanel 5.79" size''' # Resolution EPD_WIDTH = 792 EPD_HEIGHT = 272 def __init__(self): super().__init__(self.EPD_WIDTH, self.EPD_HEIGHT) self.Prepare((self.EPD_WIDTH+8) * self.EPD_HEIGHT // 8) def Prepare(self, count): '''Fill RAM and AltRAM of both chips with 0s and 1s for proper start''' self.SetRAMMP() self.SetRAMMA() self._cmd(self.SET_WRITE_RAM) self._data_s(b'\xFF' * count) self.SetRAMMA() self._cmd(self.SET_WRITE_ALTRAM) self._data_s(b'\x00' * count) self.SetRAMSP() self.SetRAMSA() self._cmd(self.SET_WRITE_RAM_SLAVE) self._data_s(b'\xFF' * count) self.SetRAMSA() self._cmd(self.SET_WRITE_ALTRAM_SLAVE) self._data_s(b'\x00' * count) def show(self, mode=1): '''Show buffer on screen. Parameters ---------- mode : int, optional 1- Fast; 2- Partial; 0- Full mode. Slowest but most clear view Raises ------ ValueError Buffer size is not as expected according to screen dimension ''' if len(self.buffer) != self.EPD_WIDTH * self.EPD_HEIGHT / 8: raise ValueError(f"Invalid frame buffer size. Expected {self.EPD_WIDTH * self.EPD_HEIGHT} bytes.") # prepare file-like object to work with bitmap_buffer = BytesIO(self.buffer) while True: chunk = bitmap_buffer.read(50) if not chunk: break self._cmd(self.SET_WRITE_RAM_SLAVE) self._data_s(chunk) # Simulate "partial data hidden" behavior (optional) bitmap_buffer.seek(-1, 1) # Not recommended for BytesIO, can cause errors # on screen intersection, half of byte is invisible. On right screen- 4 MSB's, # on right- 4 LSB's. That why need to share same byte between two screens # Read the next chunk (if any) chunk = bitmap_buffer.read(50) if not chunk: break self._cmd(self.SET_WRITE_RAM) self._data_s(chunk) bitmap_buffer.close() if mode == 1: self.FastUpdate() elif mode == 2: self.PartUpdate() else: self.Update() class Screen_420(SSD1683): '''device specifics for CrowPanel 4.2" size''' # Resolution EPD_HEIGHT = 300 EPD_WIDTH = 400 def __init__(self): raise NotImplementedError '''probably initialization may be different if screen orientation differs from 5.79 screen''' super().__init__(self.EPD_WIDTH, self.EPD_HEIGHT) self.Prepare(self.EPD_WIDTH * self.EPD_HEIGHT // 8) def Prepare(self, count): '''Fill RAM and AltRAM of chip with 0s and 1s for proper start''' self.SetRAMMP() self.SetRAMMA() self._cmd(self.SET_WRITE_RAM) self._data_s(b'\xFF' * count) self.SetRAMMA() self._cmd(self.SET_WRITE_ALTRAM) self._data_s(b'\x00' * count) def show(self, mode=1): '''Show buffer on screen. Parameters ---------- mode : int, optional 1- Fast; 2- Partial; 0- Full mode. Slowest but most clear view Raises ------ ValueError Buffer size is not as expected according to screen dimension ''' if len(self.buffer) != self.EPD_WIDTH * self.EPD_HEIGHT / 8: raise ValueError(f"Invalid frame buffer size. Expected {self.EPD_WIDTH * self.EPD_HEIGHT} bytes.") self._cmd(self.SET_WRITE_RAM) self._data_s(self.buffer) if mode == 1: self.FastUpdate() elif mode == 2: self.PartUpdate() else: self.Update() ===== Sample application ===== {{:components:elecrow_app_sample.jpg?400|}} import time time.sleep(1) # short sleep to CTRL+C if something goes wrong import CrowPanel as eink # Instantiate a Screen screen = eink.Screen_579() # Test another font # bassed on https://github.com/peterhinch/micropython-font-to-py/tree/master from writer import Writer import freesans20 wri = Writer(screen, freesans20) # prepare framebuffer screen.fill(eink.COLOR_WHITE) screen.LoadImage(30, 10, 'Images/CrowPanel_64_32.bin', 64, 32) Writer.set_textpos(screen, 130, 15) wri.printstring('CrowPanel ESP32 5.79" E-paper Display with 272*792 Resolution', True) #diagonal line, to confirm correct display between 2 screens screen.line(50, 50, 750, 222, eink.COLOR_BLACK) screen.text("diagonal line, to confirm correct", 280, 115, eink.COLOR_BLACK) screen.text("display between 2 screens", 295, 135, eink.COLOR_BLACK) # Draw arc screen.text("Draw ARC from two semi-ellipses", 30, 180, eink.COLOR_BLACK) screen.ellipse(130,170, 50, 50, eink.COLOR_BLACK, True, 3) screen.ellipse(130,175, 50, 50, eink.COLOR_WHITE, True, 3) screen.LoadImage(600, 50, 'Images/houseImg128.bin', 128, 128) screen.text("Load BW image", 615, 50, eink.COLOR_BLACK) Writer.set_textpos(screen, 280, 250) wri.printstring('Inverted Color of another font') #Load buffer to screen and display screen.show()