Add pico display feedback
This commit is contained in:
@@ -2,6 +2,9 @@ import wifi_client
|
|||||||
import rate
|
import rate
|
||||||
|
|
||||||
|
|
||||||
|
from machine import I2C, Pin
|
||||||
|
from ssd1306 import SSD1306_I2C
|
||||||
|
|
||||||
# set SSID and PASSWORD variables to the configurations of the laptop access point
|
# set SSID and PASSWORD variables to the configurations of the laptop access point
|
||||||
SSID = 'Oger-fi'
|
SSID = 'Oger-fi'
|
||||||
PASSWORD = 'Karlfreitag!'
|
PASSWORD = 'Karlfreitag!'
|
||||||
@@ -9,6 +12,10 @@ SERVER_PORT = 3000
|
|||||||
SERVER_IP_LOWER = 100
|
SERVER_IP_LOWER = 100
|
||||||
SERVER_IP_UPPER = 102
|
SERVER_IP_UPPER = 102
|
||||||
|
|
||||||
|
# Initialize I2C and OLED
|
||||||
|
i2c = I2C(0, sda=Pin(8), scl=Pin(9), freq=400000)
|
||||||
|
display = SSD1306_I2C(128, 64, i2c)
|
||||||
|
|
||||||
# connect to laptop hotspot
|
# connect to laptop hotspot
|
||||||
wifi_client.wifi_connect(SSID, PASSWORD)
|
wifi_client.wifi_connect(SSID, PASSWORD)
|
||||||
wifi_client.search_server(SERVER_IP_LOWER, SERVER_IP_UPPER, SERVER_PORT)
|
wifi_client.search_server(SERVER_IP_LOWER, SERVER_IP_UPPER, SERVER_PORT)
|
||||||
@@ -18,7 +25,7 @@ wifi_client.search_server(SERVER_IP_LOWER, SERVER_IP_UPPER, SERVER_PORT)
|
|||||||
# define Pins 0-4 for rate buttons (5 buttons)
|
# define Pins 0-4 for rate buttons (5 buttons)
|
||||||
rate_button_pins:list = [0, 1, 2, 3, 4]
|
rate_button_pins:list = [0, 1, 2, 3, 4]
|
||||||
send_button_pin:int = 6
|
send_button_pin:int = 6
|
||||||
rate.start_rating(rate_button_pins, send_button_pin)
|
rate.start_rating(rate_button_pins, send_button_pin, display)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -12,6 +12,48 @@ USER_ID = 0
|
|||||||
RATING = 0
|
RATING = 0
|
||||||
BUTTON_SEND = None
|
BUTTON_SEND = None
|
||||||
RATE_BUTTONS = []
|
RATE_BUTTONS = []
|
||||||
|
DISPLAY = None
|
||||||
|
|
||||||
|
# Helper to update display based on state
|
||||||
|
def refresh_display(message=None):
|
||||||
|
if not DISPLAY:
|
||||||
|
return
|
||||||
|
|
||||||
|
DISPLAY.fill(0)
|
||||||
|
|
||||||
|
if message:
|
||||||
|
# Show temporary message (Success, Error, Confirmed)
|
||||||
|
DISPLAY.text(message, 10, 30)
|
||||||
|
elif CURRENT_STATE == STATE_USER_ID:
|
||||||
|
# 1. User ID Mode
|
||||||
|
DISPLAY.text("Inchometer", 25, 0)
|
||||||
|
|
||||||
|
# Display binary representation: _ _ _ _ _
|
||||||
|
# Bit 4 is leftmost button, Bit 0 is rightmost
|
||||||
|
binary_str = ""
|
||||||
|
for i in range(5):
|
||||||
|
bit_pos = 4 - i
|
||||||
|
if USER_ID & (1 << bit_pos):
|
||||||
|
binary_str += "X "
|
||||||
|
else:
|
||||||
|
binary_str += "_ "
|
||||||
|
|
||||||
|
DISPLAY.text(binary_str.strip(), 30, 30)
|
||||||
|
|
||||||
|
elif CURRENT_STATE == STATE_RATING:
|
||||||
|
# 3. Rating Mode
|
||||||
|
DISPLAY.text("Select Rating", 10, 0)
|
||||||
|
|
||||||
|
if RATING == 0:
|
||||||
|
rating_text = "_"
|
||||||
|
else:
|
||||||
|
rating_text = str(RATING)
|
||||||
|
|
||||||
|
# Draw "big" text (simulated with scale if supported, or just centered)
|
||||||
|
# Standard ssd1306 doesn't have big fonts, so we just center it
|
||||||
|
DISPLAY.text(rating_text, 60, 30)
|
||||||
|
|
||||||
|
DISPLAY.show()
|
||||||
|
|
||||||
# initialize a specific Pin as Input with a pull-down resistor
|
# initialize a specific Pin as Input with a pull-down resistor
|
||||||
def initialize_pin(pin_number:int):
|
def initialize_pin(pin_number:int):
|
||||||
@@ -41,56 +83,58 @@ def rate_button_pressed(pin:Pin) -> None:
|
|||||||
return
|
return
|
||||||
|
|
||||||
if CURRENT_STATE == STATE_USER_ID:
|
if CURRENT_STATE == STATE_USER_ID:
|
||||||
# Binary input: each button toggles a bit
|
|
||||||
# Assume rightmost (highest index) is bit 0
|
|
||||||
bit_pos = len(RATE_BUTTONS) - 1 - button_index
|
bit_pos = len(RATE_BUTTONS) - 1 - button_index
|
||||||
USER_ID ^= (1 << bit_pos)
|
USER_ID ^= (1 << bit_pos)
|
||||||
print(f"Current User ID: {USER_ID} (binary: {bin(USER_ID)})")
|
print(f"Current User ID: {USER_ID}")
|
||||||
|
|
||||||
elif CURRENT_STATE == STATE_RATING:
|
elif CURRENT_STATE == STATE_RATING:
|
||||||
# Rating selection (1 to 5/6)
|
|
||||||
# Assuming the button index + 1 is the rating
|
|
||||||
RATING = button_index + 1
|
RATING = button_index + 1
|
||||||
print(f"Current Rating selected: {RATING}")
|
print(f"Current Rating selected: {RATING}")
|
||||||
|
|
||||||
# Activate send pin to confirm selection
|
refresh_display()
|
||||||
activate_sendPin()
|
activate_sendPin()
|
||||||
|
|
||||||
# Send/Confirm button is pressed
|
# Send/Confirm button is pressed
|
||||||
def sendButton_pressed(pin) -> None:
|
def sendButton_pressed(pin) -> None:
|
||||||
global CURRENT_STATE, USER_ID, RATING
|
global CURRENT_STATE, USER_ID, RATING
|
||||||
# deactivate send pin to prevent multiple sends
|
|
||||||
deactivate_sendPin()
|
deactivate_sendPin()
|
||||||
|
|
||||||
if CURRENT_STATE == STATE_USER_ID:
|
if CURRENT_STATE == STATE_USER_ID:
|
||||||
print(f"Confirmed User ID: {USER_ID}")
|
# 2. Confirmed ID display
|
||||||
|
refresh_display(f"Confirmed: {USER_ID}")
|
||||||
|
time.sleep(1)
|
||||||
|
|
||||||
CURRENT_STATE = STATE_RATING
|
CURRENT_STATE = STATE_RATING
|
||||||
print("Please select rating (1-5) and press send.")
|
refresh_display()
|
||||||
|
|
||||||
elif CURRENT_STATE == STATE_RATING:
|
elif CURRENT_STATE == STATE_RATING:
|
||||||
if RATING == 0:
|
if RATING == 0:
|
||||||
print("Please select a rating first!")
|
|
||||||
activate_sendPin()
|
activate_sendPin()
|
||||||
return
|
return
|
||||||
|
|
||||||
print(f"Confirmed Rating: {RATING}")
|
# 4. Sending / Success / Error
|
||||||
print(f"Sending request: User {USER_ID}, Rating {RATING}")
|
refresh_display("Sending...")
|
||||||
|
|
||||||
# send a request to the access point
|
success = wifi_client.send_request(USER_ID, RATING)
|
||||||
try:
|
|
||||||
wifi_client.send_request(USER_ID, RATING)
|
if success:
|
||||||
except Exception as e:
|
refresh_display("Success")
|
||||||
print(f"Error sending request: {e}")
|
else:
|
||||||
|
refresh_display("Error")
|
||||||
|
|
||||||
|
time.sleep(5) # Display status for 5 seconds
|
||||||
|
|
||||||
# Reset for next rating
|
# Reset for next rating
|
||||||
USER_ID = 0
|
USER_ID = 0
|
||||||
RATING = 0
|
RATING = 0
|
||||||
CURRENT_STATE = STATE_USER_ID
|
CURRENT_STATE = STATE_USER_ID
|
||||||
print("Done. Ready for next User ID input.")
|
refresh_display()
|
||||||
|
|
||||||
# start the rating program
|
# start the rating program
|
||||||
def start_rating(rate_button_pins:list, button_send_pin:int) -> None:
|
def start_rating(rate_button_pins:list, button_send_pin:int, oled_display=None) -> None:
|
||||||
global RATE_BUTTONS, BUTTON_SEND
|
global RATE_BUTTONS, BUTTON_SEND, DISPLAY
|
||||||
|
|
||||||
|
DISPLAY = oled_display
|
||||||
|
|
||||||
# initialize all pins as buttons and add them to the list
|
# initialize all pins as buttons and add them to the list
|
||||||
RATE_BUTTONS = []
|
RATE_BUTTONS = []
|
||||||
@@ -102,8 +146,8 @@ def start_rating(rate_button_pins:list, button_send_pin:int) -> None:
|
|||||||
# initialize send button
|
# initialize send button
|
||||||
BUTTON_SEND = initialize_pin(button_send_pin)
|
BUTTON_SEND = initialize_pin(button_send_pin)
|
||||||
|
|
||||||
print(f"System started. Pins: {rate_button_pins}, Send: {button_send_pin}")
|
print("System started.")
|
||||||
print("Please enter User ID (binary) and press send.")
|
refresh_display()
|
||||||
|
|
||||||
while True:
|
while True:
|
||||||
time.sleep(1)
|
time.sleep(1)
|
||||||
|
|||||||
158
pico_2026/ssd1306.py
Normal file
158
pico_2026/ssd1306.py
Normal file
@@ -0,0 +1,158 @@
|
|||||||
|
# MicroPython SSD1306 OLED driver, I2C and SPI interfaces
|
||||||
|
|
||||||
|
from micropython import const
|
||||||
|
import framebuf
|
||||||
|
|
||||||
|
|
||||||
|
# register definitions
|
||||||
|
SET_CONTRAST = const(0x81)
|
||||||
|
SET_ENTIRE_ON = const(0xA4)
|
||||||
|
SET_NORM_INV = const(0xA6)
|
||||||
|
SET_DISP = const(0xAE)
|
||||||
|
SET_MEM_ADDR = const(0x20)
|
||||||
|
SET_COL_ADDR = const(0x21)
|
||||||
|
SET_PAGE_ADDR = const(0x22)
|
||||||
|
SET_DISP_START_LINE = const(0x40)
|
||||||
|
SET_SEG_REMAP = const(0xA0)
|
||||||
|
SET_MUX_RATIO = const(0xA8)
|
||||||
|
SET_COM_OUT_DIR = const(0xC0)
|
||||||
|
SET_DISP_OFFSET = const(0xD3)
|
||||||
|
SET_COM_PIN_CFG = const(0xDA)
|
||||||
|
SET_DISP_CLK_DIV = const(0xD5)
|
||||||
|
SET_PRECHARGE = const(0xD9)
|
||||||
|
SET_VCOM_DESEL = const(0xDB)
|
||||||
|
SET_CHARGE_PUMP = const(0x8D)
|
||||||
|
|
||||||
|
|
||||||
|
# Subclassing FrameBuffer provides support for graphics primitives
|
||||||
|
# http://docs.micropython.org/en/latest/pyboard/library/framebuf.html
|
||||||
|
class SSD1306(framebuf.FrameBuffer):
|
||||||
|
def __init__(self, width, height, external_vcc):
|
||||||
|
self.width = width
|
||||||
|
self.height = height
|
||||||
|
self.external_vcc = external_vcc
|
||||||
|
self.pages = self.height // 8
|
||||||
|
self.buffer = bytearray(self.pages * self.width)
|
||||||
|
super().__init__(self.buffer, self.width, self.height, framebuf.MONO_VLSB)
|
||||||
|
self.init_display()
|
||||||
|
|
||||||
|
def init_display(self):
|
||||||
|
for cmd in (
|
||||||
|
SET_DISP | 0x00, # off
|
||||||
|
# address setting
|
||||||
|
SET_MEM_ADDR,
|
||||||
|
0x00, # horizontal
|
||||||
|
# resolution and layout
|
||||||
|
SET_DISP_START_LINE | 0x00,
|
||||||
|
SET_SEG_REMAP | 0x01, # column addr 127 is mapped to SEG0
|
||||||
|
SET_MUX_RATIO,
|
||||||
|
self.height - 1,
|
||||||
|
SET_COM_OUT_DIR | 0x08, # scan from COM[N] to COM0
|
||||||
|
SET_DISP_OFFSET,
|
||||||
|
0x00,
|
||||||
|
SET_COM_PIN_CFG,
|
||||||
|
0x02 if self.width > 2 * self.height else 0x12,
|
||||||
|
# timing and driving scheme
|
||||||
|
SET_DISP_CLK_DIV,
|
||||||
|
0x80,
|
||||||
|
SET_PRECHARGE,
|
||||||
|
0x22 if self.external_vcc else 0xF1,
|
||||||
|
SET_VCOM_DESEL,
|
||||||
|
0x30, # 0.83*Vcc
|
||||||
|
# display
|
||||||
|
SET_CONTRAST,
|
||||||
|
0xFF, # maximum
|
||||||
|
SET_ENTIRE_ON, # output follows RAM contents
|
||||||
|
SET_NORM_INV, # not inverted
|
||||||
|
# charge pump
|
||||||
|
SET_CHARGE_PUMP,
|
||||||
|
0x10 if self.external_vcc else 0x14,
|
||||||
|
SET_DISP | 0x01,
|
||||||
|
): # on
|
||||||
|
self.write_cmd(cmd)
|
||||||
|
self.fill(0)
|
||||||
|
self.show()
|
||||||
|
|
||||||
|
def poweroff(self):
|
||||||
|
self.write_cmd(SET_DISP | 0x00)
|
||||||
|
|
||||||
|
def poweron(self):
|
||||||
|
self.write_cmd(SET_DISP | 0x01)
|
||||||
|
|
||||||
|
def contrast(self, contrast):
|
||||||
|
self.write_cmd(SET_CONTRAST)
|
||||||
|
self.write_cmd(contrast)
|
||||||
|
|
||||||
|
def invert(self, invert):
|
||||||
|
self.write_cmd(SET_NORM_INV | (invert & 1))
|
||||||
|
|
||||||
|
def show(self):
|
||||||
|
x0 = 0
|
||||||
|
x1 = self.width - 1
|
||||||
|
if self.width == 64:
|
||||||
|
# SSD1306 only has 128x64 pixels. For 64x48 displays,
|
||||||
|
# the columns are offset by 32.
|
||||||
|
x0 += 32
|
||||||
|
x1 += 32
|
||||||
|
self.write_cmd(SET_COL_ADDR)
|
||||||
|
self.write_cmd(x0)
|
||||||
|
self.write_cmd(x1)
|
||||||
|
self.write_cmd(SET_PAGE_ADDR)
|
||||||
|
self.write_cmd(0)
|
||||||
|
self.pages - 1
|
||||||
|
self.write_cmd(self.pages - 1)
|
||||||
|
self.write_data(self.buffer)
|
||||||
|
|
||||||
|
|
||||||
|
class SSD1306_I2C(SSD1306):
|
||||||
|
def __init__(self, width, height, i2c, addr=0x3C, external_vcc=False):
|
||||||
|
self.i2c = i2c
|
||||||
|
self.addr = addr
|
||||||
|
self.temp = bytearray(2)
|
||||||
|
self.write_list = [b"\x40", None] # Co=0, D/C#=1
|
||||||
|
super().__init__(width, height, external_vcc)
|
||||||
|
|
||||||
|
def write_cmd(self, cmd):
|
||||||
|
self.temp[0] = 0x00 # Co=1, D/C#=0
|
||||||
|
self.temp[1] = cmd
|
||||||
|
self.i2c.writeto(self.addr, self.temp)
|
||||||
|
|
||||||
|
def write_data(self, buf):
|
||||||
|
self.write_list[1] = buf
|
||||||
|
self.i2c.writevto(self.addr, self.write_list)
|
||||||
|
|
||||||
|
|
||||||
|
class SSD1306_SPI(SSD1306):
|
||||||
|
def __init__(self, width, height, spi, dc, res, cs, external_vcc=False):
|
||||||
|
self.rate = 10 * 1024 * 1024
|
||||||
|
dc.init(dc.OUT, value=0)
|
||||||
|
res.init(res.OUT, value=0)
|
||||||
|
cs.init(cs.OUT, value=1)
|
||||||
|
self.spi = spi
|
||||||
|
self.dc = dc
|
||||||
|
self.res = res
|
||||||
|
self.cs = cs
|
||||||
|
import time
|
||||||
|
|
||||||
|
self.res(1)
|
||||||
|
time.sleep_ms(1)
|
||||||
|
self.res(0)
|
||||||
|
time.sleep_ms(10)
|
||||||
|
self.res(1)
|
||||||
|
super().__init__(width, height, external_vcc)
|
||||||
|
|
||||||
|
def write_cmd(self, cmd):
|
||||||
|
self.spi.init(baudrate=self.rate, polarity=0, phase=0)
|
||||||
|
self.cs(1)
|
||||||
|
self.dc(0)
|
||||||
|
self.cs(0)
|
||||||
|
self.spi.write(bytearray([cmd]))
|
||||||
|
self.cs(1)
|
||||||
|
|
||||||
|
def write_data(self, buf):
|
||||||
|
self.spi.init(baudrate=self.rate, polarity=0, phase=0)
|
||||||
|
self.cs(1)
|
||||||
|
self.dc(1)
|
||||||
|
self.cs(0)
|
||||||
|
self.spi.write(buf)
|
||||||
|
self.cs(1)
|
||||||
@@ -39,20 +39,25 @@ def wifi_connect(SSID:str, PASSWORD:str) -> None:
|
|||||||
|
|
||||||
|
|
||||||
# send a HTTP-request to the access point with the rate value
|
# send a HTTP-request to the access point with the rate value
|
||||||
def send_request(user_id:int, rating:int) -> None:
|
def send_request(user_id:int, rating:int) -> bool:
|
||||||
|
|
||||||
# create HTTP-URL
|
# create HTTP-URL
|
||||||
URL = f"http://{IP_ADRESS_SERVER}:{str(SERVER_PORT)}/ratings/{user_id}/{rating}"
|
URL = f"http://{IP_ADRESS_SERVER}:{str(SERVER_PORT)}/ratings/{user_id}/{rating}"
|
||||||
print(URL)
|
print(URL)
|
||||||
|
|
||||||
# create request
|
try:
|
||||||
response = urequests.get(URL)
|
# create request
|
||||||
|
response = urequests.get(URL)
|
||||||
|
|
||||||
# print HTTP-answer of the access point
|
# print HTTP-answer of the access point
|
||||||
print("Server response:")
|
print("Server response:")
|
||||||
print(response.text)
|
print(response.text)
|
||||||
print()
|
print()
|
||||||
response.close()
|
response.close()
|
||||||
|
return True
|
||||||
|
except Exception as e:
|
||||||
|
print(f"Request failed: {e}")
|
||||||
|
return False
|
||||||
|
|
||||||
|
|
||||||
# test, if there can be a connection to the given ip adress and port
|
# test, if there can be a connection to the given ip adress and port
|
||||||
|
|||||||
Reference in New Issue
Block a user