-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathmain.py
executable file
·204 lines (169 loc) · 6.11 KB
/
main.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
import time
import board
import busio
import adafruit_sgp30
import neopixel
import touchio
from analogio import AnalogIn
from helpers import *
PIXEL_PIN = board.D11
NUM_PIXELS = 8
# pin with touch sensitivity
TOUCH_PIN = board.A0
# pin with battery voltage for checking battery level
VBATPIN = board.D9
DISPLAY_BRIGHTNESS = 1
# number of entries to average to get the displayed value
# the higher this is, the slower and smoother color transitions will be
# 60-80 seems to be good for nice slow transitions while staying responsive
BLUR_AMOUNT = 25
# LOWBAT_PULSE_SPEED = 200
# LOWBAT_PULSE_DEPTH = 200
# voltages at which low battery mode is turned off or on when battery voltage is
# ascending or descending
# VBAT_DESCENT_THRESHOLD = 3.35
# VBAT_ASCENT_THRESHOLD = 3.45
# in each mode, how much to dim the pixels and how many of the pixels to light
DEFAULT_DIM_FACTOR = 1 # don't dim in mode 0
DEFAULT_DIM_MAPPING = [1,1,1,1,1,1,1,1]
FIRST_DIM_FACTOR = 1 # don't dim
FIRST_DIM_MAPPING = [0,1,0,0,1,0,0,1]
SECOND_DIM_FACTOR = 20 # dim a lot mode 2
SECOND_DIM_MAPPING = [1,0,0,0,0,0,0,0]
UBER_DIM_FACTOR = 30
# set up pixel array
pixels = neopixel.NeoPixel(PIXEL_PIN, NUM_PIXELS, brightness=DISPLAY_BRIGHTNESS, auto_write=True)
# set up touch sensitivity
touch_pad = TOUCH_PIN
touch = touchio.TouchIn(touch_pad)
# VBatAnalogIn = AnalogIn(VBATPIN)
# def getBatteryVoltage():
# return getVoltage(VBatAnalogIn) * 2
# Turn off board neopixel
boardPixel = neopixel.NeoPixel(board.NEOPIXEL, 1, auto_write=True)
boardPixel[0] = 0
# Set up I2C for CO2 sensor
i2c = busio.I2C(board.SCL, board.SDA, frequency=100000)
# Create library object on our I2C port
sgp30 = adafruit_sgp30.Adafruit_SGP30(i2c)
print("SGP30 serial #", [hex(i) for i in sgp30.serial])
sgp30.iaq_init()
sgp30.set_iaq_baseline(0x8973, 0x8aae)
# define an average function that calculates the average of our values
def avg(lst):
return sum(lst)/BLUR_AMOUNT
# returns a value on the color wheel described in the wheel() function
# based a given CO2 level
def getWheelValueFromCO2(CO2):
if inRange(CO2, 400,450):
return mapFromTo(CO2, 400,450, 210,255)
elif inRange(CO2, 450,600):
return mapFromTo(CO2, 450,600, 0,55)
elif inRange(CO2, 600,1300):
return mapFromTo(CO2, 600,1300, 55,85)
elif inRange(CO2, 1300,2500):
return mapFromTo(CO2, 1300,2500, 85,95)
elif inRange(CO2, 2500,5000):
return mapFromTo(CO2, 2500,5000, 95,120)
else:
return 120
# returns a color
def getColorFromCO2(CO2):
return wheel(getWheelValueFromCO2(CO2))
# shows the color, lighting the pixels corresponding to 1s in a binary array
def showColor(color, mapping=[1,1,1,1,1,1,1,1]):
# if in UberDIM mode, turn off all pixels but light the board pixel
if(mapping == "uberdim"):
pixels.fill(0)
boardPixel[0] = color
elif mapping == 'off':
pixels.fill(0)
boardPixel[0] = 0
else:
# turn off the board pixel
boardPixel[0] = 0
# crawl the mapping and light up the pixels which should be activated
for n,m in enumerate(mapping):
if m == 1:
pixels[n] = color
else:
pixels[n] = 0
# runs a startup animation
# lights turn on from bottom to top, with a slight color change.
def startupAnimation(delay, destinationColor, colorStep):
# start out with all pixels off
animationMapping = [0,0,0,0,0,0,0,0]
i = 0
# start the color the right amount away from destination
color = destinationColor + colorStep * 8
for i in range(8):
animationMapping[i] = 1
showColor(wheel(color), animationMapping)
color -= colorStep
time.sleep(delay)
# set up list of last values to update and average
vals = [400 for x in range(BLUR_AMOUNT)]
# batLow = False
tick = int(time.monotonic())
touched = False
mode = 0
# show startup animation
startupAnimation(.045, 210, 3)
while True:
# update which second are we on, and save the last one
lastTick = [tick][0]
tick = int(time.monotonic())
# check touches and cycle through brightness modes if new touch is detected
if touch.value:
if touched == False:
# activate touch functions
# print("TOUCHED!")
# increment mode
mode = increment(mode)
touched = True
else:
touched = False
# grab the carbon dioxide and TVOC levels from the sensor
# eCO2, TVOC = sgp30.eCO2, sgp30.TVOC
eCO2 = sgp30.eCO2
# only do the following every 10 seconds (on rising edge of 10th second)
# if tick % 10 == 0 and lastTick %10 != 0:
# # get the battery voltage
# voltage = getBatteryVoltage()
# print(voltage)
# if batLow == False and voltage <= VBAT_DESCENT_THRESHOLD:
# batLow = True
# elif batLow == True and voltage >= VBAT_ASCENT_THRESHOLD:
# batLow = False
# remove oldest datapoint from our list of values and add the new datapoint
vals.remove(vals[0])
vals.append(eCO2)
# the value we'll work with is the average of our list of values
blurredVal = avg(vals)
# these print statements are usefull, but slow down the program if it's not
# plugged in and able to actually print to serial
# print(vals)
# print("eCO2 = %d ppm" % (sgp30.eCO2))
# print(blurredVal)
# print(voltage)
c = getColorFromCO2(blurredVal)
if mode == 0:
displayColor = c # don't dim
pixelMapping = DEFAULT_DIM_MAPPING
elif mode == 1:
displayColor = dim(c, FIRST_DIM_FACTOR)
pixelMapping = FIRST_DIM_MAPPING
elif mode == 2:
displayColor = dim(c, SECOND_DIM_FACTOR) # SuperDim for nighttime :)
pixelMapping = SECOND_DIM_MAPPING
elif mode == 3:
displayColor = dim(c, UBER_DIM_FACTOR) # UberDIM !! >:}
pixelMapping = "uberdim"
elif mode == 4:
pixelMapping = 'off'
# # pulse the display color if in low battery mode
# if batLow == True:
# ShowColor(pulse(displayColor, LOWBAT_PULSE_SPEED, LOWBAT_PULSE_DEPTH), pixelMapping)
# # otherwise just display the color
# else:
showColor(displayColor, pixelMapping)